【转】来自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浅析及应用。分享给大家供大家参考。具体分析如下: 余弦相似性 原理:首先我们先把两段文本分词,列出来所有单词,其次我们计算每个词语的词频,最后把词语...
在这个项目中,我们关注的是PHP实现的一种基于余弦定理的分词计算文本相似度的方法。余弦定理通常被用于解决几何问题,但在这里,它被巧妙地应用于文本分析。 首先,我们要理解什么是余弦定理。在平面几何中,余弦...
在文本查重工具中,模板模式可能被用来定义基本的查重流程,如读取文件、分词、计算相似度等,而具体的查重算法(如余弦定理或simhash)则由不同的子类实现。 其次,**策略模式** 允许在运行时选择算法或策略。在...
7. **基于Meaning的文本相似度计算**:通常使用TF-IDF、余弦相似度或Jaccard相似度等方法来衡量文本的意义相似度,帮助识别和分类主题。 8. **文本分类**:除了以上提到的贝叶斯和决策树,还有其他算法如支持向量机...
1. KNN文本分类算法:KNN是一种基于实例的学习,通过计算待分类文本与训练集中已知类别的文本之间的相似度(如余弦相似度),选择最接近的K个邻居,根据邻居的类别分布决定待分类文本的类别。 2. 特征选择:为了降低...
在实际应用中,通常只关注正值,因为负值在文本相似度计算中没有意义。 在找出相似文章的场景中,首先使用TF-IDF算法对每篇文章进行关键词提取,确保选取的关键词能反映文章的主题。然后,将所有文章的关键词合并,...
3. 余弦定理与文本相似度计算 余弦定理是一种数学工具,用于计算两个向量的夹角的余弦值。在文本处理中,它可以用来衡量两个文档(文本)在向量空间模型中的相似度。简历管理系统利用余弦定理计算简历文本之间的...
相似度通常通过向量间的夹角余弦来度量,余弦值越大,相似度越高。 3. 计算每个类别在K个邻居中的权重,权重等于属于该类别的训练样本与测试样本的相似度之和。 4. 将新文本分配到权重最高的类别中。K值的选择会影响...
3. 在训练集中找出与新文本最相似的K个文本,这里的相似度通常用向量夹角余弦来度量。 4. 对于新文本的K个邻居,计算每类的权重,权重是邻居中属于该类的训练样本与测试样本的相似度之和。 5. 将新文本分配到权重...
在文本分析、推荐系统等领域,余弦相似度常用来评估文档或用户兴趣的相似性,因为它忽略了向量的长度,只关注方向。 **主成分分析**是一种降维技术,旨在找到原始特征的线性组合,使新特征具有最大的方差,同时尽...
在本项目中,我们将探讨VSM的实现细节,包括文档表示、相似度计算以及使用的两种主要方法——余弦相似度(Cosine Similarity)和TF-IDF算法。 首先,让我们了解VSM的基本概念。在VSM中,每个文档被视为一个向量,这...
在本课程中,我们将深入探讨一系列相关技术,包括单词和句子的表示方法、文本相似度计算、算法应用以及语言模型。 首先,单词表示是自然语言处理的基础,常见的方法有 Bag-of-Words 模型。这种模型通过将每个单词...
相似度通常使用欧氏距离或余弦相似度等度量。在文本分类中,我们可以将文档表示为词频向量,然后应用KNN来找到最近的邻居。Java中可以使用诸如Weka这样的机器学习库来实现KNN算法。 2. Naive Bayes(朴素贝叶斯)...
- **距离度量**:定义一个合适的距离函数,如欧氏距离、曼哈顿距离或余弦相似度,来衡量样本之间的相似性。 - **K值选择**:选择合适的K值,通常通过交叉验证来确定。 - **分类**:计算新样本与所有训练样本的距离,...
在文本分类中,KNN通常通过计算文本之间的相似度来确定邻居,如TF-IDF(词频-逆文档频率)向量距离或余弦相似度。 2. SVM(支持向量机): SVM是一种二分类模型,它的基本模型是定义在特征空间上的间隔最大的线性...
距离计算通常使用欧氏距离、曼哈顿距离或余弦相似度等方法。 7. K均值(K-Means) K均值是聚类算法,通过迭代将数据点分配到最近的K个中心(初始随机选择),然后更新中心为该类数据点的均值,直至聚类不再变化。...
在文本分类中,通过计算文档向量与各个类别向量的余弦相似度,可以找出与文档最相似的类别,从而完成分类。余弦相似度的值介于-1和1之间,值越接近1表示两个向量越相似。 "单词字典树"(如Trie或Patricia Trie)是...
11. **余弦定理**:在新闻分类中,余弦相似度用于衡量两个向量(如文档向量)之间的角度,从而判断它们的主题相似度。 12. **信息指纹**:为文本或多媒体内容创建独特标识,用于版权保护、抄袭检测等。 13. **数学...