基础知识

文本相似度计算是把文本投影到向量空间,文本的相似度是把文本投影到向量空间,用向量相似度来表示语义相似度,通过比较计算向量的空间距离来比较文本的相似度。

Jaccard系数

Jaccard系数是计算两个集合重合度的常用方法:两个集合A和B交集元素的个数在A、B并集中所占的比例,称为这两个集合的Jaccard系数,用符号 J(A,B) 表示。Jaccard系数是衡量两个集合相似度的一种指标,公式如下:
在这里插入图片描述
其中,A,B集合中的元素为是文本中的全部词项,且集合中相同词项只保留一个。

余弦相似度

余弦相似度,是通过计算两个向量的夹角余弦值来评估向量间的相似度。将向量根据坐标映射到向量空间。求得他们的夹角,并得出夹角对应的余弦值,余弦值就可以用来表示这两个向量的相似性。
夹角越小,余弦值越接近于1,它们的方向更加吻合,则越相似。两个向量A,B之间的余弦相似度计算公式如下:
在这里插入图片描述

计算文本之间的余弦相似度时,首先要对文本分词、去除停用词,构建所有文本的词袋,然后计算文本词项的tf-idf值,接着利用词袋模型将每个文本都映射到统一的向量空间中,向量的坐标值就是对应的特征词项的tf-idf值。

实验内容

  • 选定3个文本:doc1,doc2,doc3,其中doc1和doc2是两篇关于“机器学习”的介绍,doc3是1篇人大会议报道。
  • 计算上述3个文档之间的Jaccard系数矩阵(行列为文档,元素值为两个文档之间的系数值)。
  • 利用向量空间模型计算词项-文档关联矩阵,并计算3个文档之间余弦相似度矩阵(行列为文档,元素值为两个文档之间的相似度值)。
  • 对比分析Jaccard系数和余弦相似度的性能。

步骤

  1. 获取停用词表,文本分词
    在这里插入图片描述
    这一步是自然语言处理的基本操作,代码略。
  2. 获取去停用词后的词典
    在这里插入图片描述
def get_dic(text_list, stop_words_list):
    dic = {}
    for myword in text_list:
        if not (myword.strip() in stop_words_list):
            dic.setdefault(myword, 0)
            dic[myword] += 1
    return dic

dic为该文本去除停用词之后的词典,键是文本中的词,值是这个词出现的频数

  1. 计算余弦相似度
    首先,调用步骤1中的函数,清洗文本,获取全部词项
def get_cossimi_matrix(corpus_path, stop_path):
    all = []
    catelist = os.listdir(corpus_path)
    stop_list = get_stop_list(stop_path)
        for mydir in catelist:
        print('   %s  ' % mydir, end='')
        text_path = corpus_path + '/' + mydir
        samp_seg_list = get_seg_list(text_path)  # 样例文本分词后的list
        samp_str = ''
        for word in samp_seg_list:
            if word not in stop_list:
                samp_str += word
                samp_str += ' '
        all.append(samp_str)

然后,计算每一个单词的tf-idf值,将文本用tf-idf矩阵表示,将文本映射到向量空间

    vectorizer = CountVectorizer() 
    transformer = TfidfTransformer()  # 统计每个词语的tf-idf权值函数
    X = vectorizer.fit_transform(all)  # 将文本转为词频矩阵
    tfidf = transformer.fit_transform(X)  # 计算tf-idf值
    # 将tf-idf矩阵抽取出来,元素a[i][j]表示词项j在文本i中的tf-idf权重, 每一行表示一个文本在总的向量空间模型中映射的向量
    tfidf_matrix = tfidf.toarray()

接着,编写计算余弦相似度函数

def get_cossimi(x, y):
    myx = np.array(x)
    myy = np.array(y)
    cos1 = np.sum(myx * myy)
    cos21 = np.sqrt(sum(myx * myx))
    cos22 = np.sqrt(sum(myy * myy))
    return cos1/float(cos21 * cos22)#return: 两个向量夹角的余弦值

传入参数为上一步求得的用tf-idf表示的文本,返回值是两个文本的余弦相似度。
最后,调用余弦相似度计算函数,按照列表循环的方式,分别比较三个文本的相似度

print('\n\t\tML_1\t\t\tML_2\t\t\trenda')
    cossimi_matrix = [[0 for i in range(len(tfidf_matrix))] for i in range(len(tfidf_matrix))]
    for i in range(0, len(tfidf_matrix)):#i元素为三个文本
        print('text_%d\t' % (i + 1), end='')
        for j in range(0, len(tfidf_matrix)):#j为被比较的文本,也是这三个文本,和i顺序相同。
            cossimi_matrix[i][j] = get_cossimi(tfidf_matrix[i], tfidf_matrix[j])#计算余弦相似度
            if i == j:
                print("1.000000\t\t", end=' ')#如果i,j是同一篇文本,则余弦相似度为1
            else:
                print("%f\t\t"%get_cossimi(tfidf_matrix[i], tfidf_matrix[j]), end=' ')

采用嵌套循环的方式,两两比较文本相似度:

1)如果是同一篇文本,那么余弦值为1,输出余弦值
2)如果是不同文本,则调用余弦相似度计算函数,输出余弦值
因为本文语料是三篇文本,所以该程序一共输出3*3=9次。

  1. 计算Jaccard系数
    首先,清洗文本,获取每篇文档的文本字典
    fstop_seg_list = get_stop_list(stop_path)

    dics = []
    text_name = []
    catelist = os.listdir(corpus_path)
    for mydir in catelist:
        text_path = corpus_path + '/'+ mydir
        seg_list = get_seg_list(text_path)  # 样例文本分词后的list
        # 获取文本的去除停用词之后的字典
        dics.append(get_dic(seg_list, fstop_seg_list))
        text_name.append(mydir)
        print('   %s  '%mydir, end='')

这里,字典dics获取的是来自上文中统计好的去除停用词后的词典。
接着,编写计算jaccard系数函数

def get_jaccard(x, y):
    all_words = {}  # 大字典
    all_words.update(x)  # 用字典sample_words更新大字典
    all_words.update(y)  # 用字典mytest1_words更新大字典

    xORy = len(all_words) #x和y并集中元素的个数
    xANDy = 0 # x和y交集中元素的个数
    for key in x.keys():
        for key2 in y.keys():
            if key == key2:
                xANDy += 1

    simi = float(xANDy)/xORy
    return simi

在这个函数中,传入参数是用词典表示的文本,返回值是两个文本之间的Jaccard系数。

最后,采用嵌套循环的方式,两两计算Jaccard系数。

jaccard_matrix = [[0 for i in range(len(dics))] for i in range(len(dics))]
    for i in range(0, len(dics)):
        print('text_%d\t'%(i+1), end='')
        for j in range(0, len(dics)):
            jaccard_matrix[i][j] = get_jaccard(dics[i], dics[j])
            print('%f\t\t'%get_jaccard(dics[i], dics[j]), end='')
        print('\n')
    return jaccard_matrix

结果及分析

实验结果如下:
余弦相似度矩阵为:
在这里插入图片描述
text_1和ML_1,text_2和ML_2,text_3和renda为同一篇文档。
两篇机器学习文档的相似度为:0.146312
机器学习文档与人大会议文档的相似度为:0.0044870.014308.
可以看出,两篇相同主题的文档相似度较高,且远远高于不同主题的文档。
Jaccard系数矩阵为:
在这里插入图片描述
两篇机器学习文档的相似度为:0.091429
机器学习文档与人大会议文档的相似度为:0.0090190.009231.

对比余弦相似度矩阵,综合来看,余弦相似度较jaccard系数在衡量文本相似度时区分度更高,效果更明显。

分析:

这是由于mjaccard系数在计算时,不考虑词项在文档中出现的次数,也不考虑罕见词可能带来更大的信息量这个问题。而向量空间模型兼顾到了这两点,因而余弦相似度取得了更好的效果。


版权声明:本文为yang_live原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
原文链接:https://blog.csdn.net/yang_live/article/details/105079425