• 设为首页
  • 点击收藏
  • 手机版
    手机扫一扫访问
    迪恩网络手机版
  • 关注官方公众号
    微信扫一扫关注
    迪恩网络公众号

R语言实现46种距离算法

原作者: [db:作者] 来自: [db:来源] 收藏 邀请

作者:张丹,R语言中文社区专栏特邀作者,《R的极客理想》系列图书作者,民生银行大数据中心数据分析师,前况客创始人兼CTO。
个人博客 http://fens.me, Alexa全球排名70k。


前言

距离算法是做数据挖掘常用的一类算法,距离算法有很多种,比如欧式距离、马氏距离、皮尔逊距离,距离算法主要应用在计算数据集之间关系。本文用R语言来philentropy包,实现多种距离的算法,很多可能是大家完全没有听过的,让我们在开拓一下知识领域吧。


目录

  1. 距离算法包philentropy

  2. 46种距离算法详解

  3. 距离函数的使用


1

距离算法包philentropy

在做距离算法调研时,无意中发了philentropy包。它实现了46个不同距离算法和相似性度量,通过不同数据的相似度比较,为基础研究提供了科学基础。philentropy包,为聚类、分类、统计推断、拟合优度、非参数统计、信息理论和机器学习提供了核心的计算框架,支持基于单变量或者多变量的概率函数的计算。

philentropy包主要包括了2种度量的计算方法,距离度量和信息度量。本文介绍距离度量的使用,对于信息度量的使用,请耐心等待下一篇:R语言实现信息度量

philentropy项目github地址:https://github.com/HajkD/philentropy

本文的系统环境为:

  • Win10 64bit

  • R: 3.4.2 x86_64-w64-mingw32

安装philentropy包,非常简单,一条命令就可以了。

1~ R
2> install.packages("philentropy")
3> library(philentropy)


查看距离算法列表

 1> getDistMethods()
2 [1"euclidean"         "manhattan"         "minkowski"         "chebyshev"        
3 [5"sorensen"          "gower"             "soergel"           "kulczynski_d"     
4 [9"canberra"          "lorentzian"        "intersection"      "non-intersection" 
5[13"wavehedges"        "czekanowski"       "motyka"            "kulczynski_s"     
6[17"tanimoto"          "ruzicka"           "inner_product"     "harmonic_mean"    
7[21"cosine"            "hassebrook"        "jaccard"           "dice"             
8[25"fidelity"          "bhattacharyya"     "hellinger"         "matusita"         
9[29"squared_chord"     "squared_euclidean" "pearson"           "neyman"           
10[33"squared_chi"       "prob_symm"         "divergence"        "clark"            
11[37"additive_symm"     "kullback-leibler"  "jeffreys"          "k_divergence"     
12[41"topsoe"            "jensen-shannon"    "jensen_difference" "taneja"           
13[45"kumar-johnson"     "avg"    


46个距离算法,有一些是我们常用的比如:euclidean,manhattan,minkowski,pearson, cosine,squared_chi, 其他的我也不知道,正好拓宽知识,好好学习一下。

philentropy包的函数,其实很简单,只有14个,大量的算法其实都已经被封装到distance()函数中,直接使用distance()函数就行完成各种算法的计算,让我们使用起来会非常方便。我们来看一下,函数列表:

 1distance(): 计算距离
2getDistMethods(),获得距离算法列表
3dist.diversity(),概率密度函数之间的距离差异
4estimate.probability(),从计数向量估计概率向量
5lin.cor(),线性相关性判断
6H(): 香农熵, Shannon’s Entropy H(X)
7JE() : 联合熵, Joint-Entropy H(X,Y)
8CE() : 条件熵, Conditional-Entropy H(X|Y)
9MI() : 互信息, Shannon’s Mutual Information I(X,Y)
10KL() : KL散度, Kullback–Leibler Divergence
11JSD() : JS散度,Jensen-Shannon Divergence
12gJSD() : 通用JS散度,Generalized Jensen-Shannon Divergence
13binned.kernel.est(),实现了KernSmooth包提供的核密度估计函数的接口


从函数列表来看,主要分为3种类别的函数:

  • 第一类是距离测量的函数,包括

    distance(),getDistMethods(), dist.diversity(), lin.cor()和 estimate.probability()。

  • 第二类是相关性分析,包括lin.cor()函数。

  • 第三类是信息度量函数

    H(),JE(),CE(),MI(),KL(),JSD(),gJSD()。


2

46种算法详解

接下来,就让我们深入每个算法吧,从名字到公式,再到函数使用,最后到使用场景。

距离算法列表:

 1euclidean:欧式距离,是一个通常采用的距离定义,在m维空间中两个点之间的真实距离,或者向量的自然长度(即该点到原点的距离)。
2manhattan:曼哈顿距离,用于几何空间度量,表示两个点在标准坐标系上的绝对轴距距离总和。
3minkowski:闵可夫斯基距离,是欧氏空间中的广义距离函数,其参数p值的不同代表着对空间不同的度量。
4chebyshev:切比雪夫距离,是向量空间中的一种度量,二个点之间的距离定义是其各坐标数值差绝对值的最大值。
5sorensen:测量每个样本单位,对单位总数的距离测量贡献度,广告用于生态学。
6gower:高尔距离,将向量空间缩放为规范化空间,可计算逻辑值,数字,文本的距离,距离结果为01之间的数字。
7soergel:测量每个样本单位,对最大值总数的距离测量贡献度。
8kulczynski:与soergel相反,测量每个样本单位,对最小值总数的距离测量贡献度。
9canberra:堪培拉距离,是矢量空间中的点对之间的距离的数值度量,它是L_1距离的加权版本。
10lorentzian:洛伦兹距离,绝对的差异并应用自然对数。
11intersection:交叉距离,最小轨道交叉距离,是天文学中用于评估天文物体之间潜在的近距离接近和碰撞风险的度量。它被定义为两个物体的密切轨道的最近点之间的距离。
12non-intersection:非交叉距离
13wavehedges:波浪距离,
14czekanowski:
15motyka:莫蒂卡方程,是czekanowski的一半。
16kulczynski_s:
17tanimoto:是标准化内积的另一种变体。
18ruzicka:
19inner_product:内部产品空间,计算两个向量的内积产生标量,有时称为标量积或点积。
20harmonic_mean:调和平均值。
21cosine:余弦距离,是用向量空间中两个向量夹角的余弦值作为衡量两个个体间差异的大小的度量。
22hassebrook(PCE):利用P•Q来测量能量的峰值,简称PCE。
23jaccard:杰卡德距离,用于计算样本间的相似度,分子是A和B的交集大小,分母是A和B的并集大小。
24dice:骰子
25fidelity:保真度,在量子信息理论中,保真度是两个量子态“接近”的度量。
26bhattacharyya:巴氏距离,测量两个概率分布的相似性。
27hellinger:海林格,用来度量两个概率分布的相似度,它是F散度的一种。
28matusita:
29squared_chord:
30squared_euclidean:欧式距离的平方
31pearson:皮尔森距离,分子是两个集合的交集大小,分母是两个集合大小的几何平均值,是余弦距离的一种变型。
32neyman:奈曼,
33squared_chi:
34prob_symm:
35divergence:散度,
36clark:克拉克,
37additive_symm:算术和几何平均散度.
38kullback-leibler:KL散度,用于计算相熵或信息偏差,是衡量两个分布(P、Q)之间的距离,越小越相似。
39jeffreys:杰弗里斯,J分歧。
40k_divergence:K散度,
41topsoe:托普索,是k_divergence加法的对称形式。
42jensen-shannon:詹森香农,是topsoe距离的一半。
43jensen_difference:
44taneja:塔内加,计算算术和几何平均偏差。
45kumar-johnson:库马尔-约翰逊,
46avg:平均


由于精力和基础知识有限,对每一种算法还没有更深入的理解和使用,后面会继续补充。这些距离的详细解释,请参考文章 http://csis.pace.edu/ctappert/dps/d861-12/session4-p2.pdf

这么多种的距离算法,其实可以分成8大距离家族,每个家族中不同的算法思路是类似的,可以通过变形或参数不同赋值,进行算法的相互转换。

L_p Minkowski家族,通过对Minkowski 算法p值的不同赋值,可以转换成不同的算法,当p=1时Minkowski距离转为曼哈顿距离;当p=2变Minkowski距离转为欧氏距离;当p接近极限最大值时,Minkowski距离是转为切比雪夫距离。

L_1家族,用于准确的测量绝对差异的特征。

Intersection 交叉距离家族,用于交叉点之间的相似度变换。

Inter Product 家族,几何空间的相似性度量,用于特定的 P•Q 变量来计算。

Squared-chord 家族,在量子信息理论中,量子态“接近”的度量。

Squared L_2 家族( X^2 Squared 家族),以平方的欧几里得距离做为被除数。

香浓信息熵家族,信息熵偏差测量。

组合公式,利用多种算法思路,进行组合的距离测量方法。


3

距离函数的使用

了解了这么多的距离算法后,让我们来使用一下philentropy包强大的功能函数,把算法落地。


3.1 distance()函数的使用

distance()函数,用来计算两个概率密度函数之间的距离和相似度,上面所列出的所有的距离算法都被封装在了这个函数里。

distance()函数定义:

1distance(x, method = "euclidean", p = NULL, test.na = TRUE, unit = "log", est.prob = NULL)


参数列表:

  • x, 数值类型的向量或数据集

  • method, 算法的名称

  • p, minkowski闵可夫斯基距离的p值,p=1为曼哈顿距离,p=2为欧氏距离,p取极限时是切比雪夫距离

  • test.na, 检测数据集是否有NA值,不检测为FALSE,计算会快。

  • unit,对数化的单位,依赖于日志计算的距离

  • est.prob 从计数估计概率,默认值为NULL


计算euclidean距离,用iris的数据集。

 1library(magrittr)
2
3# 查看iris数据集
4head(iris)
5  Sepal.Length Sepal.Width Petal.Length Petal.Width Species
61          5.1         3.5          1.4         0.2  setosa
72          4.9         3.0          1.4         0.2  setosa
83          4.7         3.2          1.3         0.2  setosa
94          4.6         3.1          1.5         0.2  setosa
105          5.0         3.6          1.4         0.2  setosa
116          5.4         3.9          1.7         0.4  setosa


计算第1个点(第1行)和第2个点(第2行)的euclidean距离,分别使用philentropy包的distance(),Stats包的dist(),和自己通过公式计算。

 1# 使用distance()函数
2> dat1<-iris[1:2,-5]
3> distance(dat1, method="euclidean")
4Metric: 'euclidean' using unit: 'log'.
5euclidean 
60.5385165 
7
8# 再使用系统自带的dist()函数
9> dist(dat1)
10          1
112 0.5385165
12
13# 公式计算
14> (dat1[1,]-dat1[2,])^2 %>% sum %>% sqrt
15[10.5385165


3种方法,计算的结果是完全一致。

接下来,我们构建一个iris的距离矩阵,为了展示清楚,我们选iris的前6个点来计算距离。分别使用distance()和dist()函数。

 1> dat2<-head(iris[,-5])
2
3# 距离矩阵
4> distance(dat2)
5Metric: 'euclidean' using unit: 'log'.
6          v1        v2       v3        v4        v5        v6
7v1 0.0000000 0.5385165 0.509902 0.6480741 0.1414214 0.6164414
8v2 0.5385165 0.0000000 0.300000 0.3316625 0.6082763 1.0908712
9v3 0.5099020 0.3000000 0.000000 0.2449490 0.5099020 1.0862780
10v4 0.6480741 0.3316625 0.244949 0.0000000 0.6480741 1.1661904
11v5 0.1414214 0.6082763 0.509902 0.6480741 0.0000000 0.6164414
12v6 0.6164414 1.0908712 1.086278 1.1661904 0.6164414 0.0000000
13
14# 下三角距离矩阵
15> dist(dat2)
16          1         2         3         4         5
172 0.5385165                                        
183 0.5099020 0.3000000                              
194 0.6480741 0.3316625 0.2449490                    
205 0.1414214 0.6082763 0.5099020 0.6480741          
216 0.6164414 1.0908712 1.0862780 1.1661904 0.6164414


验证后,我们就可以放心使用distance()函数。通过对比实验,我们可以很快的学习并使用各种距离算法。


3.2 dist.diversity()函数

dist.diversity()函数,用来计算所有距离的值。由于有一些距离有对于数据集本身的要求,所以我们需要构建一个能适应所有距离算法的数据集。

 1# 生成数据集,2个点,10个维度
2> P <- 1:10/sum(1:10)
3> Q <- 20:29/sum(20:29)
4> x <- rbind(P,Q)
5
6# 打印数据集
7> head(x)
8        [,1]       [,2]       [,3]       [,4]       [,5]      [,6]      [,7]      [,8]
9P 0.01818182 0.03636364 0.05454545 0.07272727 0.09090909 0.1090909 0.1272727 0.1454545
10Q 0.08163265 0.08571429 0.08979592 0.09387755 0.09795918 0.1020408 0.1061224 0.1102041
11       [,9]     [,10]
12P 0.1636364 0.1818182
13Q 0.1142857 0.1183673


使用dist.diversity()函数计算所有的距离。

 1> dist.diversity(x,p=2)
2euclidean         manhattan         minkowski         chebyshev          sorensen 
3       0.12807130        0.35250464        0.12807130        0.06345083        0.17625232 
4            gower           soergel      kulczynski_d          canberra        lorentzian 
5       0.03525046        0.29968454        0.42792793        2.09927095        0.34457827 
6     intersection  non-intersection        wavehedges       czekanowski            motyka 
7       0.82374768        0.17625232        3.16657887        0.17625232        0.58812616 
8     kulczynski_s          tanimoto           ruzicka     inner_product     harmonic_mean 
9       2.33684211        0.29968454        0.70031546        0.10612245        0.94948528 
10           cosine        hassebrook           jaccard              dice          fidelity 
11       0.93427641        0.86613103        0.13386897        0.07173611        0.97312397 
12    bhattacharyya         hellinger          matusita     squared_chord squared_euclidean 
13       0.02724379        0.32787819        0.23184489        0.05375205        0.01640226 
14          pearson            neyman       squared

鲜花

握手

雷人

路过

鸡蛋
该文章已有0人参与评论

请发表评论

全部评论

专题导读
上一篇:
R语言中使用subset函数对数据进行分类管理操作发布时间:2022-07-18
下一篇:
Python和R语言之间的抉择(二)发布时间:2022-07-18
热门推荐
热门话题
阅读排行榜

扫描微信二维码

查看手机版网站

随时了解更新最新资讯

139-2527-9053

在线客服(服务时间 9:00~18:00)

在线QQ客服
地址:深圳市南山区西丽大学城创智工业园
电邮:jeky_zhao#qq.com
移动电话:139-2527-9053

Powered by 互联科技 X3.4© 2001-2213 极客世界.|Sitemap