`
yuanzhen
  • 浏览: 31937 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

实现文本相似度算法(余弦定理

    博客分类:
  • java
阅读更多
【转】来自http://my.oschina.net/BreathL/blog/42477

Lucene中的评分机制,也是算一个相似度的问题,不过它采用的是计算向量间的夹角(余弦公式),在google黑板报中的:数学之美(余弦定理和新闻分类) 也有说明,可以通过余弦定理来判断相似度;于是决定自己动手试试。

       首相选择向量的模型:在以字为向量还是以词为向量的问题上,纠结了一会;后来还是觉得用字,虽然词更为准确,但分词却需要增加额外的复杂度,并且此项目要求速度,准确率可以放低,于是还是选择字为向量。

       然后每个字在章节中出现的次数,便是以此字向量的值。现在我们假设:

       章节1中出现的字为:Z1c1,Z1c2,Z1c3,Z1c4……Z1cn;它们在章节中的个数为:Z1n1,Z1n2,Z1n3……Z1nm;

       章节2中出现的字为:Z2c1,Z2c2,Z2c3,Z2c4……Z2cn;它们在章节中的个数为:Z2n1,Z2n2,Z2n3……Z2nm;

       其中,Z1c1和Z2c1表示两个文本中同一个字,Z1n1和Z2n1是它们分别对应的个数,

       最后我们的相似度可以这么计算:

       程序实现如下:(若有可优化或更好的实现请不吝赐教)




package test;

import java.io.UnsupportedEncodingException;
import java.util.Date;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;

public class CosineSimilarAlgorithm {

	public static void main(String[] args) {
		String s1="将两个字符串中的中文字符以及出现的总数封装到,AlgorithmMap中  ";
		String s2="根据输入的Unicode字符,获取它的GB2312编码或者ascii编码";
		long t1=new Date().getTime();
		double d=getSimilarity(s1, s2);
		long t2=new Date().getTime();
		System.out.println("相似度:"+d+"————计算时间:"+(t2-t1)+"毫秒");
	}
	
	
	public static double getSimilarity(String doc1, String doc2) {  
         if (doc1 != null && doc1.trim().length() > 0 && doc2 != null&& doc2.trim().length() > 0) {  
            Map<Integer, int[]> AlgorithmMap = new HashMap<Integer, int[]>();  
            //将两个字符串中的中文字符以及出现的总数封装到,AlgorithmMap中  
            for (int i = 0; i < doc1.length(); i++) {  

                 char d1 = doc1.charAt(i);  
                 if(isHanZi(d1)){  
                    int charIndex = getGB2312Id(d1);  //ch在GB2312中的位置
                     if(charIndex != -1){  
                         int[] fq = AlgorithmMap.get(charIndex);  
                        if(fq != null && fq.length == 2){  
                            fq[0]++;  
                         }else {  
                             fq = new int[2];  
                            fq[0] = 1;  
                             fq[1] = 0;  
                            AlgorithmMap.put(charIndex, fq);  
                         }  
                    }  
                 }  
            }  

    

             for (int i = 0; i < doc2.length(); i++) {  

                 char d2 = doc2.charAt(i);  

                if(isHanZi(d2)){  
                    int charIndex = getGB2312Id(d2);  
                    if(charIndex != -1){  
                        int[] fq = AlgorithmMap.get(charIndex);
                         if(fq != null && fq.length == 2){ 
                             fq[1]++; 
                        }else {  
                            fq = new int[2];  
                            fq[0] = 0;  
                            fq[1] = 1;  
                            AlgorithmMap.put(charIndex, fq);  
                         }  
                     }  
                 }  
             }  
             Iterator<Integer> iterator = AlgorithmMap.keySet().iterator();  
             double sqdoc1 = 0;  
            double sqdoc2 = 0;  
             double denominator = 0; 
             while(iterator.hasNext()){  
                 int[] c = AlgorithmMap.get(iterator.next());  
                 denominator += c[0]*c[1];  
                 sqdoc1 += c[0]*c[0];  
                 sqdoc2 += c[1]*c[1];  
            }  
             return denominator / Math.sqrt(sqdoc1*sqdoc2);  
        } else {  
             throw new NullPointerException(
                    " the Document is null or have not cahrs!!");  
         }  
     }  

   

     public static boolean isHanZi(char ch) {  
         // 判断是否汉字  
        return (ch >= 0x4E00 && ch <= 0x9FA5);  
     }  

     /**  
     * 根据输入的Unicode字符,获取它的GB2312编码或者ascii编码,  
     * @param ch  
     *  输入的GB2312中文字符或者ASCII字符(128个) 
     * @return ch在GB2312中的位置,-1表示该字符不认识  
     */ 

     public static short getGB2312Id(char ch) {  
        try {  
            byte[] buffer = Character.toString(ch).getBytes("GB2312");  
             if (buffer.length != 2) {  
                 // 正常情况下buffer应该是两个字节,否则说明ch不属于GB2312编码,故返回'?',此时说明不认识该字符  
                 return -1;  
             }  
             int b0 = (int) (buffer[0] & 0x0FF) - 161; // 编码从A1开始,因此减去0xA1=161  
             int b1 = (int) (buffer[1] & 0x0FF) - 161; // 第一个字符和最后一个字符没有汉字,因此每个区只收16*6-2=94个汉字  
            return (short) (b0 * 94 + b1);  

         } catch (UnsupportedEncodingException e) {  
             e.printStackTrace();  

         }  
         return -1;  
     }  

} 






感觉相似度0.98以上的可以用,呵呵,速度还不错
分享到:
评论

相关推荐

    .NET下文本相似度算法余弦定理和SimHash浅析及应用实例分析

    本文实例讲述了.NET下文本相似度算法余弦定理和SimHash浅析及应用。分享给大家供大家参考。具体分析如下: 余弦相似性 原理:首先我们先把两段文本分词,列出来所有单词,其次我们计算每个词语的词频,最后把词语...

    text-similarity-php, 通过余弦定理 分词计算文本相似度PHP版.zip

    在这个项目中,我们关注的是PHP实现的一种基于余弦定理的分词计算文本相似度的方法。余弦定理通常被用于解决几何问题,但在这里,它被巧妙地应用于文本分析。 首先,我们要理解什么是余弦定理。在平面几何中,余弦...

    java文本查重工具类封装

    在文本查重工具中,模板模式可能被用来定义基本的查重流程,如读取文件、分词、计算相似度等,而具体的查重算法(如余弦定理或simhash)则由不同的子类实现。 其次,**策略模式** 允许在运行时选择算法或策略。在...

    文本挖掘算法总结.doc

    7. **基于Meaning的文本相似度计算**:通常使用TF-IDF、余弦相似度或Jaccard相似度等方法来衡量文本的意义相似度,帮助识别和分类主题。 8. **文本分类**:除了以上提到的贝叶斯和决策树,还有其他算法如支持向量机...

    数据挖掘中的文本挖掘的分类算法综述.doc

    1. KNN文本分类算法:KNN是一种基于实例的学习,通过计算待分类文本与训练集中已知类别的文本之间的相似度(如余弦相似度),选择最接近的K个邻居,根据邻居的类别分布决定待分类文本的类别。 2. 特征选择:为了降低...

    TF-IDF与余弦相似性的应用(二) 找出相似文章

    在实际应用中,通常只关注正值,因为负值在文本相似度计算中没有意义。 在找出相似文章的场景中,首先使用TF-IDF算法对每篇文章进行关键词提取,确保选取的关键词能反映文章的主题。然后,将所有文章的关键词合并,...

    毕业设计论文简历管理系统.pdf

    3. 余弦定理与文本相似度计算 余弦定理是一种数学工具,用于计算两个向量的夹角的余弦值。在文本处理中,它可以用来衡量两个文档(文本)在向量空间模型中的相似度。简历管理系统利用余弦定理计算简历文本之间的...

    数据挖掘-基于贝叶斯算法及KNN算法.docx

    相似度通常通过向量间的夹角余弦来度量,余弦值越大,相似度越高。 3. 计算每个类别在K个邻居中的权重,权重等于属于该类别的训练样本与测试样本的相似度之和。 4. 将新文本分配到权重最高的类别中。K值的选择会影响...

    数据挖掘-基于贝叶斯算法及KNN算法.pdf

    3. 在训练集中找出与新文本最相似的K个文本,这里的相似度通常用向量夹角余弦来度量。 4. 对于新文本的K个邻居,计算每类的权重,权重是邻居中属于该类的训练样本与测试样本的相似度之和。 5. 将新文本分配到权重...

    Unit-15:余弦相似度,主成分分析

    在文本分析、推荐系统等领域,余弦相似度常用来评估文档或用户兴趣的相似性,因为它忽略了向量的长度,只关注方向。 **主成分分析**是一种降维技术,旨在找到原始特征的线性组合,使新特征具有最大的方差,同时尽...

    VSM信息检索模型(向量空间模型)

    在本项目中,我们将探讨VSM的实现细节,包括文档表示、相似度计算以及使用的两种主要方法——余弦相似度(Cosine Similarity)和TF-IDF算法。 首先,让我们了解VSM的基本概念。在VSM中,每个文档被视为一个向量,这...

    机器学习-文本分析-1

    在本课程中,我们将深入探讨一系列相关技术,包括单词和句子的表示方法、文本相似度计算、算法应用以及语言模型。 首先,单词表示是自然语言处理的基础,常见的方法有 Bag-of-Words 模型。这种模型通过将每个单词...

    基于tags的文本分类,使用KNN, Naive Bayes, Softmax(使用java).zip

    相似度通常使用欧氏距离或余弦相似度等度量。在文本分类中,我们可以将文档表示为词频向量,然后应用KNN来找到最近的邻居。Java中可以使用诸如Weka这样的机器学习库来实现KNN算法。 2. Naive Bayes(朴素贝叶斯)...

    code_贝叶斯算法_KNN分类_

    - **距离度量**:定义一个合适的距离函数,如欧氏距离、曼哈顿距离或余弦相似度,来衡量样本之间的相似性。 - **K值选择**:选择合适的K值,通常通过交叉验证来确定。 - **分类**:计算新样本与所有训练样本的距离,...

    文本分类器,KNN,SVM,贝叶斯等都有

    在文本分类中,KNN通常通过计算文本之间的相似度来确定邻居,如TF-IDF(词频-逆文档频率)向量距离或余弦相似度。 2. SVM(支持向量机): SVM是一种二分类模型,它的基本模型是定义在特征空间上的间隔最大的线性...

    人工智能的常用十种算法(1).pdf

    距离计算通常使用欧氏距离、曼哈顿距离或余弦相似度等方法。 7. K均值(K-Means) K均值是聚类算法,通过迭代将数据点分配到最近的K个中心(初始随机选择),然后更新中心为该类数据点的均值,直至聚类不再变化。...

    classifier4j jar包下载

    在文本分类中,通过计算文档向量与各个类别向量的余弦相似度,可以找出与文档最相似的类别,从而完成分类。余弦相似度的值介于-1和1之间,值越接近1表示两个向量越相似。 "单词字典树"(如Trie或Patricia Trie)是...

    谷歌黑板报-数学之美 数学在信息检索和自然语言处理中的主导作用和奇妙应用 共45页.pdf

    11. **余弦定理**:在新闻分类中,余弦相似度用于衡量两个向量(如文档向量)之间的角度,从而判断它们的主题相似度。 12. **信息指纹**:为文本或多媒体内容创建独特标识,用于版权保护、抄袭检测等。 13. **数学...

Global site tag (gtag.js) - Google Analytics