李航《统计学习方法》第2版 第10章 HMM实现分词(代码实现)

利用HMM模型实现分词:
四种状态:
B:词语的开头
M:一个词语的中间词
E:一个词语的结果
S:非词语,单个词
学习:Baum-Welch算法
预测:维特比算法
数据集:人民日报1998年中文标注语料库
链接:https://pan.baidu.com/s/1SKi9DUjxuh6tENfm6jmNCA
提取码:hz3q
复制这段内容后打开百度网盘手机App,操作更方便哦

代码引用自:www.pkudodo.com

#coding=utf-8
#Author:Dodo
#Date:2018-12-10
#Email:[email protected]
#Blog:www.pkudodo.com
'''
学习:Baum-Welch算法
预测:维特比算法
数据集:人民日报1998年中文标注语料库
------------------------------
运行结果:
-------------------原文----------------------
深圳有个打工者阅览室
去年12月,我在广东深圳市出差,听说南山区工商分局为打工者建了个免费图书阅览室,这件新鲜事引起了我的兴趣。
12月18日下午,我来到了这个阅览室。阅览室位于桂庙,临南油大道,是一间轻体房,面积约有40平方米,内部装修得整洁干净,四周的书架上摆满了书,并按政治、哲学、法律法规、文化教育、经济、科技、艺术、中国文学、外国文学等分类,屋中央有两排书架,上面也摆满了图书和杂志。一些打工青年或站或蹲,认真地阅读,不时有人到借阅台前办理借书或还书手续。南山区在深圳市西边,地处城乡结合部,外来打工者较多。去年2月,南山区工商分局局长王安全发现分局对面的公园里常有不少打工者业余时间闲逛,有时还滋扰生事。为了给这些打工者提供一个充实自己的场所,他提议由全分局工作人员捐款,兴建一个免费阅览室。领导带头,群众响应,大家捐款1.4万元,购买了近千册图书。3月6日,建在南头繁华的南新路和金鸡路交叉口的阅览室开放了。从此,这里每天都吸引了众多借书、看书的人们,其中不仅有打工者,还有机关干部、公司职员和个体户。到了夏天,由于阅览室所在地被工程征用,南山区工商分局便把阅览室迁到了桂庙。阅览室的管理人员是两名青年,男的叫张攀,女的叫赵阳。张攀自己就是湖北来的打工者,听说南山区工商分局办免费阅览室,便主动应聘来服务。阅览室每天从早9时开到晚10时,夜里张攀就住在这里。他谈起阅览室里的图书,翻着一本本的借阅名册,如数家珍,对图书和工作的挚爱之情溢于言表。我在这里碰到南山区华英大厦一位叫聂煜的女青年,她说她也是个打工者,由于春节探家回来后就要去市内工作,很留恋这里的这个免费阅览室,想抓紧时间多看些书,她还把自己买的几本杂志捐给了阅览室。在阅览室的捐书登记簿上,记录着这样的数字:工商系统内部捐书3550册,社会各界捐书250册。我在阅览室读到了这样几封感谢信:深圳瑞兴光学厂的王志明写道:“我们这些年轻人远离了家乡,来到繁华紧张的都市打工,辛劳之余,能有机会看书读报,感到特别充实。”深圳文光灯泡厂的江虹说:“南山区工商分局的干部职工捐款、捐书,给我们打工者提供良好的学习环境,鼓励我们求知上进,真是办了一件大好事,他们是我们打工者的知音。”(本报记者罗华)
-------------------分词后----------------------
深圳|有个|打|工者|阅览室
去年|12月|,|我|在|广东|深圳|市出|差|,|听|说|南|山区|工商|分局|为|打|工者|建了|个|免费|图书|阅览室|,|这件|新|鲜事|引起|了|我|的|兴趣|。
12月|18日|下午|,|我来|到|了|这个|阅览室|。|阅览|室位|于|桂庙|,|临南油|大道|,|是|一间|轻|体房|,|面积|约|有|40|平方|米|,|内|部装|修得|整洁|干净|,|四|周|的|书架|上|摆满|了|书|,|并|按|政治|、|哲学|、|法律|法规|、|文化|教育|、|经济|、|科技|、|艺术|、|中国|文学|、|外国|文学|等|分类|,|屋|中央|有|两排|书架|,|上面|也|摆满|了|图书|和|杂志|。|一些|打工|青年|或站|或|蹲|,|认真|地阅|读|,|不时|有|人到|借阅|台前|办理|借书|或|还书|手续|。|南|山区|在|深圳|市|西边|,|地处|城乡|结合部|,|外来|打|工者|较|多|。|去年|2月|,|南|山区|工商|分局|局长|王|安全|发现|分局|对面|的|公园|里|常有|不少|打|工者|业余|时间|闲逛|,|有时|还|滋扰|生事|。|为|了|给|这些|打|工者|提供|一个|充实|自己|的|场所|,|他|提议|由全|分局|工作|人员|捐款|,|兴建|一个|免费|阅览室|。|领导|带头|,|群众|响应|,|大家|捐款|1.4万|元|,|购买|了|近|千册|图书|。|3月|6日|,|建在|南头|繁华|的|南|新路|和|金鸡|路交|叉口|的|阅览室|开放|了|。|从此|,|这里|每天|都|吸引|了|众多|借书|、|看书|的|人们|,|其中|不仅|有|打|工者|,|还|有|机关|干部|、|公司|职员|和|个|体户|。|到|了|夏天|,|由于|阅览室|所|在|地|被|工程|征用|,|南|山区|工商|分局|便|把|阅览|室迁|到|了|桂庙|。|阅览室|的|管理|人员|是|两|名|青年|,|男|的|叫|张|攀|,|女|的|叫|赵阳|。|张|攀|自己|就|是|湖北|来|的|打|工者|,|听|说|南|山区|工商|分局|办|免费|阅览室|,|便|主动|应|聘来|服务|。|阅览室|每天|从|早9时|开到|晚|10|时|,|夜里|张|攀|就|住|在|这里|。|他谈|起|阅览|室里|的|图书|,|翻着|一|本本|的|借阅|名册|,|如数|家珍|,|对|图书|和|工作|的|挚爱|之|情溢|于|言表|。|我|在|这里|碰到|南|山区|华英|大厦|一位|叫|聂|煜|的|女|青年|,|她|说|她|也|是|个|打|工者|,|由于|春节|探家|回来|后|就|要|去市|内|工作|,|很|留恋|这里|的|这个|免费|阅览室|,|想|抓紧|时间|多|看些|书|,|她|还|把|自己|买|的|几本|杂志|捐给|了|阅览室|。|在|阅览室|的|捐书|登|记簿|上|,|记录|着|这样|的|数字|:|工商|系统|内部|捐书|3550册|,|社会|各界|捐书|250册|。|我|在|阅览室|读到|了|这样|几|封感|谢信|:|深圳|瑞兴|光学|厂|的|王|志明|写道|:|“|我们|这些|年|轻人|远离|了|家乡|,|来到|繁华|紧张|的|都|市|打工|,|辛劳|之余|,|能|有|机会|看书|读报|,|感到|特别|充实|。|”|深圳|文光|灯|泡厂|的|江虹|说|:|“|南|山区|工商|分局|的|干部|职工|捐款|、|捐书|,|给|我们|打|工者|提供|良好|的|学习|环境|,|鼓励|我们|求知|上进|,|真是|办|了|一件|大好|事|,|他们|是|我们|打|工者|的|知音|。|”|(|本报|记者|罗华|)

运行时长:3.6s
'''
import numpy as np
import time

def trainParameter(fileName):
    '''
    依据训练文本统计PI、A、B
    :param fileName: 训练文本
    :return: 三个参数
    '''
    #定义一个查询字典,用于映射四种标记在数组中对应的位置,方便查询
    # B:词语的开头
    # M:一个词语的中间词
    # E:一个词语的结果
    # S:非词语,单个词
    statuDict = {
    
    'B':0, 'M':1, 'E':2, 'S':3}

    #每个字只有四种状态,所以下方的各类初始化中大小的参数均为4
    #初始化PI的一维数组,因为对应四种状态,大小为4
    PI = np.zeros(4)
    #初始化状态转移矩阵A,涉及到四种状态各自到四种状态的转移,因为大小为4x4
    A = np.zeros((4, 4))
    #初始化观测概率矩阵,分别为四种状态到每个字的发射概率
    #因为是中文分词,使用ord(汉字)即可找到其对应编码,这里用一个65536的空间来保证对于所有的汉字都能
    #找到对应的位置来存储
    B = np.zeros((4, 65536))
    #去读训练文本
    fr = open(fileName, encoding='utf-8')

    #文本中的每一行认为是一个训练样本
    #在统计上,三个参数依据“10.3.2” Baum-Welch算法内描述的统计
    #PI依据式10.35
    #A依据10.37
    #B依据10.38
    #注:并没有使用Baum-Welch算法,只是借助了其内部的三个参数生成公式,其实
    #公式并不是Baum-Welch特有的,只是在那一节正好有描述
    for line in fr.readlines():
        #---------------------训练集单行样例--------------------
        #深圳  有  个  打工者  阅览室
        #------------------------------------------------------
        #可以看到训练样本已经分词完毕,词语之间空格隔开,因此我们在生成统计时主要借助以下思路:
        # 1.先将句子按照空格隔开,例如例句中5个词语,隔开后变成一个长度为5的列表,每个元素为一个词语
        # 2.对每个词语长度进行判断:
        #       如果为1认为该词语是S,即单个字
        #       如果为2则第一个是B,表开头,第二个为E,表结束
        #       如果大于2,则第一个为B,最后一个为E,中间全部标为M,表中间词
        # 3.统计PI:该句第一个字的词性对应的PI中位置加1
        #           例如:PI = [0, 0, 0, 0],当本行第一个字是B,即表示开头时,PI中B对应位置为0,
        #               则PI = [1, 0, 0, 0],全部统计结束后,按照计数值再除以总数得到概率
        #   统计A:对状态链中位置t和t-1的状态进行统计,在矩阵中相应位置加1,全部结束后生成概率
        #   统计B:对于每个字的状态以及字内容,生成状态到字的发射计数,全部结束后生成概率
        #   注:可以看一下“10.1.1 隐马尔可夫模型的定义”一节中三个参数的定义,会有更清晰一点的认识
        #-------------------------------------------------------
        #对单行句子按空格进行切割
        curLine = line.strip().split()
        #对词性的标记放在该列表中
        wordLabel = []
        #对每一个单词进行遍历
        for i in range(len(curLine)):
            #如果长度为1,则直接将该字标记为S,即单个词
            if len(curLine[i]) == 1:
                label = 'S'
            else:
                #如果长度不为1,开头为B,最后为E,中间添加长度-2个M
                #如果长度刚好为2,长度-2=0也就不添加了,反之添加对应个数的M
                label = 'B' + 'M' * (len(curLine[i]) - 2) + 'E'

            #如果是单行开头第一个字,PI中对应位置加1,
            if i == 0: PI[statuDict[label[0]]] += 1

            #对于该单词中的每一个字,在生成的状态链中统计B
            for j in range(len(label)):
                #遍历状态链中每一个状态,并找到对应的中文汉字,在B中
                #对应位置加1
                B[statuDict[label[j]]][ord(curLine[i][j])] += 1

            #在整行的状态链中添加该单词的状态链
            #注意:extend表直接在原先元素的后方添加,
            #可以百度一下extend和append的区别
            wordLabel.extend(label)

        #单行所有单词都结束后,统计A信息
        #因为A涉及到前一个状态,因此需要等整条状态链都生成了才能开始统计
        for i in range(1, len(wordLabel)):
            #统计t时刻状态和t-1时刻状态的所有状态组合的出现次数
            A[statuDict[wordLabel[i - 1]]][statuDict[wordLabel[i]]] += 1

    #上面代码在统计上全部是统计的次数,实际运算需要使用概率,
    #下方代码是将三个参数的次数转换为概率
    #----------------------------------------
    #对PI求和,概率生成中的分母
    sum = np.sum(PI)
    #遍历PI中每一个元素,元素出现的次数/总次数即为概率
    for i in range(len(PI)):
        #如果某元素没有出现过,该位置为0,在后续的计算中这是不被允许的
        #比如说某个汉字在训练集中没有出现过,那在后续不同概率相乘中只要有
        #一项为0,其他都是0了,此外整条链很长的情况下,太多0-1的概率相乘
        #不管怎样最后的结果都会很小,很容易下溢出
        #所以在概率上我们习惯将其转换为log对数形式,这在书上是没有讲的
        #x大的时候,log也大,x小的时候,log也相应小,我们最后比较的是不同
        #概率的大小,所以使用log没有问题

        #那么当单向概率为0的时候,log没有定义,因此需要单独判断
        #如果该项为0,则手动赋予一个极小值
        if PI[i] == 0:  PI[i] = -3.14e+100
        #如果不为0,则计算概率,再对概率求log
        else: PI[i] = np.log(PI[i] / sum)

    #与上方PI思路一样,求得A的概率对数
    for i in range(len(A)):
        sum = np.sum(A[i])
        for j in range(len(A[i])):
            if A[i][j] == 0: A[i][j] = -3.14e+100
            else: A[i][j] = np.log(A[i][j] / sum)

    #与上方PI思路一样,求得B的概率对数
    for i in range(len(B)):
        sum = np.sum(B[i])
        for j in range(len(B[i])):
            if B[i][j] == 0: B[i][j] = -3.14e+100
            else:B[i][j] = np.log(B[i][j] / sum)

    #返回统计得到的三个参数
    return PI, A, B

def loadArticle(fileName):
    '''
    加载文章
    :param fileName:文件路径
    :return: 文章内容
    '''
    #初始化文章列表
    artical = []
    #打开文件
    fr = open(fileName, encoding='utf-8')
    #按行读取文件
    for line in fr.readlines():
        #读到的每行最后都有一个\n,使用strip将最后的回车符去掉
        line = line.strip()
        #将该行放入文章列表中
        artical.append(line)
    #将文章返回
    return artical

def participle(artical, PI, A, B):
    '''
    分词
    算法依据“10.4.2 维特比算法”
    :param artical:要分词的文章
    :param PI: 初始状态概率向量PI
    :param A: 状态转移矩阵
    :param B: 观测概率矩阵
    :return: 分词后的文章
    '''
    #初始化分词后的文章列表
    retArtical = []

    #对文章按行读取
    for line in artical:
        #初始化δ,δ存放四种状态的概率值,因为状态链中每个状态都有
        #四种概率值,因此长度时该行的长度
        delta = [[0 for i in range(4)] for i in range(len(line))]
        #依据算法10.5 第一步:初始化
        for i in range(4):
            #初始化δ状态链中第一个状态的四种状态概率
            delta[0][i] = PI[i] + B[i][ord(line[0])]
        #初始化ψ,初始时为0
        psi = [[0 for i in range(4)] for i in range(len(line))]

        #算法10.5中的第二步:递推
        #for循环的符号与书中公式一致,可以对比着看来理解
        #依次处理整条链
        for t in range(1, len(line)):
            #对于链中的米格状态,求四种状态概率
            for i in range(4):
                #初始化一个临时列表,用于存放四种概率
                tmpDelta = [0] * 4
                for j in range(4):
                    # 计算第二步中的δ,该部分只计算max内部,不涉及后面的bi(o)
                    # 计算得到四个结果以后,再去求那个max即可
                    # 注:bi(Ot)并不在max的式子中,是求出max以后再乘b的
                    #   此外读者可能注意到书中的乘法在这里变成了加法,这是由于原先是概率
                    #   直接相乘,但我们在求得概率时,同时取了log,取完log以后,概率的乘法
                    #   也就转换为加法了,同时也简化了运算
                    #   所以log优点还是很多的对不?
                    tmpDelta[j] = delta[t - 1][j] + A[j][i]

                #找到最大的那个δ * a,
                maxDelta = max(tmpDelta)
                #记录最大值对应的状态
                maxDeltaIndex = tmpDelta.index(maxDelta)

                #将找到的最大值乘以b放入,
                #注意:这里同样因为log变成了加法
                delta[t][i] = maxDelta + B[i][ord(line[t])]
                #在ψ中记录对应的最大状态索引
                psi[t][i] = maxDeltaIndex

        #建立一个状态链列表,开始生成状态链
        sequence = []
        #算法10.5 第三步:终止
        #在上面for循环全部结束后,很明显就到了第三步了
        #获取最后一个状态的最大状态概率对应的索引
        i_opt = delta[len(line) - 1].index(max(delta[len(line) - 1]))
        #在状态链中添加索引
        #注:状态链应该是B、M、E、S,这里图方便用了0、1、2、3,其实一样的
        sequence.append(i_opt)
        #算法10.5 第四步:最优路径回溯
        #从后往前遍历整条链
        for t in range(len(line) - 1, 0, -1):
            #不断地从当前时刻t的ψ列表中读取到t-1的最优状态
            i_opt = psi[t][i_opt]
            #将状态放入列表中
            sequence.append(i_opt)
        #因为是从后往前将状态放入的列表,所以这里需要翻转一下,变成了从前往后
        sequence.reverse()

        #开始对该行分词
        curLine = ''
        #遍历该行每一个字
        for i in range(len(line)):
            #在列表中放入该字
            curLine += line[i]
            #如果该字是3:S->单个词  或  2:E->结尾词 ,则在该字后面加上分隔符 |
            #此外如果改行的最后一个字了,也就不需要加 |
            if (sequence[i] == 3 or sequence[i] == 2) and i != (len(line) - 1):
                curLine += '|'
        #在返回列表中添加分词后的该行
        retArtical.append(curLine)
    #返回分词后的文章
    return retArtical

if __name__ == '__main__':
    # 开始时间
    start = time.time()

    #依据现有训练集统计PI、A、B
    PI, A, B = trainParameter('HMMTrainSet.txt')

    #读取测试文章
    artical = loadArticle('testArtical.txt')

    #打印原文
    print('-------------------原文----------------------')
    for line in artical:
        print(line)

    #进行分词
    partiArtical = participle(artical, PI, A, B)

    #打印分词结果
    print('-------------------分词后----------------------')
    for line in partiArtical:
        print(line)

    #结束时间
    print('time span:', time.time() - start)

猜你喜欢

转载自blog.csdn.net/weixin_43646592/article/details/113064681