zoukankan      html  css  js  c++  java
  • 基于Python经典版成语接龙逻辑实现

      基于Python经典版成语接龙逻辑实现

      今天,想检验一下前期Python学习成功,加上好友提起斗一波成语接龙,为完全碾压,轻松取胜,便顺手写一个适合日常群聊PK的成语接龙小程序。具体实现流程如下:

      成语俗语词库下载

      前往搜狗输入法官网,找到词库页面,搜索成语,定位进入 成语俗语 页面, 下载 官方词库 -- 成语俗语【官方推荐】.scel

      搜狗成语俗语词库转txt备用

      github上有现成的scel转txt的代码,此处个人调试为适用python3.6的版本

    import struct
    import os
    
    # 搜狗的scel词库就是保存的文本的unicode编码,每两个字节一个字符(中文汉字或者英文字母)
    # 找出其每部分的偏移位置即可
    # 主要两部分
    # 1.全局拼音表,貌似是所有的拼音组合,字典序
    #       格式为(index,len,pinyin)的列表
    #       index: 两个字节的整数 代表这个拼音的索引
    #       len: 两个字节的整数 拼音的字节长度
    #       pinyin: 当前的拼音,每个字符两个字节,总长len
    #
    # 2.汉语词组表
    #       格式为(same,py_table_len,py_table,{word_len,word,ext_len,ext})的一个列表
    #       same: 两个字节 整数 同音词数量
    #       py_table_len:  两个字节 整数
    #       py_table: 整数列表,每个整数两个字节,每个整数代表一个拼音的索引
    #
    #       word_len:两个字节 整数 代表中文词组字节数长度
    #       word: 中文词组,每个中文汉字两个字节,总长度word_len
    #       ext_len: 两个字节 整数 代表扩展信息的长度,好像都是10
    #       ext: 扩展信息 前两个字节是一个整数(不知道是不是词频) 后八个字节全是0
    #
    #      {word_len,word,ext_len,ext} 一共重复same次 同音词 相同拼音表
    
    
    # 拼音表偏移,
    startPy = 0x1540;
    
    # 汉语词组表偏移
    startChinese = 0x2628;
    
    # 全局拼音表
    GPy_Table = {}
    
    # 解析结果
    # 元组(词频,拼音,中文词组)的列表
    GTable = []
    
    # 原始字节码转为字符串
    def byte2str(data):
        pos = 0
        str = ''
        while pos < len(data):
            c = chr(struct.unpack('H', bytes([data[pos], data[pos + 1]]))[0])
            if c != chr(0):
                str += c
            pos += 2
        return str
    
    # 获取拼音表
    def getPyTable(data):
        data = data[4:]
        pos = 0
        while pos < len(data):
            index = struct.unpack('H', bytes([data[pos],data[pos + 1]]))[0]
            pos += 2
            lenPy = struct.unpack('H', bytes([data[pos], data[pos + 1]]))[0]
            pos += 2
            py = byte2str(data[pos:pos + lenPy])
    
            GPy_Table[index] = py
            pos += lenPy
    
    # 获取一个词组的拼音
    def getWordPy(data):
        pos = 0
        ret = ''
        while pos < len(data):
            index = struct.unpack('H', bytes([data[pos], data[pos + 1]]))[0]
            ret += GPy_Table[index]
            pos += 2
        return ret
    
    # 读取中文表
    def getChinese(data):
        pos = 0
        while pos < len(data):
            # 同音词数量
            same = struct.unpack('H', bytes([data[pos], data[pos + 1]]))[0]
    
            # 拼音索引表长度
            pos += 2
            py_table_len = struct.unpack('H', bytes([data[pos], data[pos + 1]]))[0]
    
            # 拼音索引表
            pos += 2
            py = getWordPy(data[pos: pos + py_table_len])
    
            # 中文词组
            pos += py_table_len
            for i in range(same):
                # 中文词组长度
                c_len = struct.unpack('H', bytes([data[pos], data[pos + 1]]))[0]
                # 中文词组
                pos += 2
                word = byte2str(data[pos: pos + c_len])
                # 扩展数据长度
                pos += c_len
                ext_len = struct.unpack('H', bytes([data[pos], data[pos + 1]]))[0]
                # 词频
                pos += 2
                count = struct.unpack('H', bytes([data[pos], data[pos + 1]]))[0]
    
                # 保存
                GTable.append((count, py, word))
    
                # 到下个词的偏移位置
                pos += ext_len
    
    
    def scel2txt(file_name):
        print('-' * 60)
        with open(file_name, 'rb') as f:
            data = f.read()
    
        print("词库名:", byte2str(data[0x130:0x338])) # .encode('GB18030')
        print("词库类型:", byte2str(data[0x338:0x540]))
        print("描述信息:", byte2str(data[0x540:0xd40]))
        print("词库示例:", byte2str(data[0xd40:startPy]))
    
        getPyTable(data[startPy:startChinese])
        getChinese(data[startChinese:])
    
    if __name__ == '__main__':
    
        # scel所在文件夹路径
        in_path = "E:python_workspace"
        # 输出词典所在文件夹路径
        out_path = "coal_dict.txt"
    
        fin = [fname for fname in os.listdir(in_path) if fname[-5:] == ".scel"]
        for f in fin:
            f = os.path.join(in_path, f)
            scel2txt(f)
    
        # 保存结果
        with open(out_path, 'w', encoding='utf8') as f:
            f.writelines([word+'
    ' for count, py, word in GTable])

      业务逻辑实现

    from IPython.core.interactiveshell import InteractiveShell
    
    InteractiveShell.ast_node_interactivity = "all"
    
    from pypinyin import pinyin, lazy_pinyin, Style
    
    idiom_dic = {}
    idiom_list = []
    idiom_char_dic = {}
    
    with open('E:python_workspacecoal_dict.txt', 'r', encoding='utf8') as r:
        for line in r:
            line = line.strip()
            if None == line or line == '':
                continue
            idiom_list.append(line)
            key = lazy_pinyin(line)[0]
            value = ''
            if key in idiom_dic.keys():
                value = idiom_dic[key] + ',' + line
            else:
                value = line
            idiom_dic[key] = value
            # 汉字接龙
            key_char = line[0]
            value_char = ''
            if key_char in idiom_char_dic.keys():
                value_char = idiom_char_dic[key_char] + ',' + line
            else:
                value_char = line
            idiom_char_dic[key_char] = value_char
    
    
    # 汉字接龙,polyphone -- 是否匹配读音
    def idiom_next_char(idiom, polyphone=False):
        if idiom not in idiom_list:
            res = idiom + ' is not one idiom'
        else:
            last = idiom[len(idiom) - 1]
            if polyphone:
                pass
                if last not in idiom_char_dic:
                    res = 'library without the supply idioms'
                else:
                    aa = idiom_char_dic[last]
                    last = lazy_pinyin(idiom)[len(idiom) - 1]
                    bb = idiom_dic[last]
                    aa_list = aa.split(',')
                    bb_list = bb.split(',')
                    cd_list = set(aa_list).intersection(set(bb_list))  # 求并集
                    res = ','.join(cd_list)
            else:
                if last not in idiom_char_dic:
                    res = 'library without the supply idioms'
                else:
                    res = idiom_char_dic[last]
        return res
    
    
    # 拼音接龙
    def idiom_next(idiom):
        if idiom not in idiom_list:
            res = idiom + ' is not one idiom'
        else:
            last = lazy_pinyin(idiom)[len(idiom) - 1]
            if last not in idiom_dic:
                res = 'library without the supply idioms'
            else:
                res = idiom_dic[last]
        return res
    
    
    # print(idiom_next('怒发冲冠'))
    
    # 汉字定长接龙
    def idiom_multi_char_length(idiom, length=10, polyphone=False):
        index = 0
        res_list = [idiom]
        while index < length:
            res = idiom_next_char(idiom, polyphone)
            if 'idiom' in res:
                break
            else:
                res_next = res.split(',')
                idiom = res_next[0]
            res_list.append(idiom)
            index = index + 1
        return res_list
    
    
    # 拼音定长接龙
    def idiom_multi_length(idiom, length=10):
        index = 0
        res_list = [idiom]
        while index < length:
            res = idiom_next(idiom)
            if 'idiom' in res:
                break
            else:
                res_next = res.split(',')
                idiom = res_next[0]
            res_list.append(idiom)
            index = index + 1
        return res_list
    
    
    def check_none_follow_list():
        none_follow = []
        for idiom in idiom_list:
            res = idiom_next(idiom)
            if 'idiom' in res:
                none_follow.append(idiom)
        return none_follow
    
    
    # none_supply = check_none_follow_list()
    # print(none_supply)

      简单调试

      pycharm 2018.2版本依赖,支持 运行调试功能,即即直接加载 .py 代码到内存,可以直接调用声明的函数和变量,打开方式:Run/Debug Configurations -- Configuration -- Execution -- Run with Python console。

      这样调试尤其便利:

      1、成语接龙(读音匹配)

    idiom_multi_length('刻舟求剑')
    Out[6]: 
    ['刻舟求剑',
     '兼爱无私',
     '死败涂地',
     '低昂不就',
     '灸艾分痛',
     '同胞共气',
     '期月有成',
     '撑岸就船',
     '传杯换盏',
     '粘皮带骨',
     '告朔饩羊']

      2、成语接龙(汉字匹配)

    idiom_multi_char_length('刻舟求剑')
    Out[5]: 
    ['刻舟求剑',
     '剑拔弩张',
     '张本继末',
     '末大必折',
     '折本买卖',
     '卖卜测字',
     '字挟风霜',
     '霜刀小径',
     '径情而行',
     '行行出状元',
     '元恶大憝']

      3、成语接龙(汉字、读音匹配)

    idiom_multi_char_length('刻舟求剑',polyphone=True)
    Out[7]: 
    ['刻舟求剑',
     '剑拔弩张',
     '张冠李戴',
     '戴凭夺席',
     '席丰履厚',
     '厚貌深情',
     '情恕理遣',
     '遣将调兵',
     '兵来将挡水来土掩',
     '掩耳不闻',
     '闻弦歌而知雅意']
  • 相关阅读:
    iOS之UITableView的上拉刷新
    iOS xml文件的解析方式 XMLDictionary,GDataXMLNode,NSXMLParser
    iOS学习基本常识
    iOS常用宏定义
    iOS查错机制
    轻量级sqlite是增删改查
    iOS开发UI篇—ios应用数据存储方式(归档) :转发
    iOS面向对象的建模:MVC(OC基础)
    iOS下bound,center和frame
    CSS----学习2
  • 原文地址:https://www.cnblogs.com/nyatom/p/10913307.html
Copyright © 2011-2022 走看看