python實(shí)現(xiàn)層次聚類的方法
層次聚類算法
顧名思義,層次聚類就是一層一層的進(jìn)行聚類,可以由上向下把大的類別(cluster)分割,叫作分裂法;也可以由下向上對(duì)小的類別進(jìn)行聚合,叫作凝聚法;但是一般用的比較多的是由下向上的凝聚方法。
分裂法:
分裂法指的是初始時(shí)將所有的樣本歸為一個(gè)類簇,然后依據(jù)某種準(zhǔn)則進(jìn)行逐漸的分裂,直到達(dá)到某種條件或者達(dá)到設(shè)定的分類數(shù)目。用算法描述:
輸入:樣本集合D,聚類數(shù)目或者某個(gè)條件(一般是樣本距離的閾值,這樣就可不設(shè)置聚類數(shù)目)
輸出:聚類結(jié)果
1.將樣本集中的所有的樣本歸為一個(gè)類簇;
repeat:
2.在同一個(gè)類簇(計(jì)為c)中計(jì)算兩兩樣本之間的距離,找出距離最遠(yuǎn)的兩個(gè)樣本a,b;
3.將樣本a,b分配到不同的類簇c1和c2中;
4.計(jì)算原類簇(c)中剩余的其他樣本點(diǎn)和a,b的距離,若是dis(a)<dis(b),則將樣本點(diǎn)歸到c1中,否則歸到c2中;
util: 達(dá)到聚類的數(shù)目或者達(dá)到設(shè)定的條件
凝聚法:
凝聚法指的是初始時(shí)將每個(gè)樣本點(diǎn)當(dāng)做一個(gè)類簇,所以原始類簇的大小等于樣本點(diǎn)的個(gè)數(shù),然后依據(jù)某種準(zhǔn)則合并這些初始的類簇,直到達(dá)到某種條件或者達(dá)到設(shè)定的分類數(shù)目。用算法描述:
輸入:樣本集合D,聚類數(shù)目或者某個(gè)條件(一般是樣本距離的閾值,這樣就可不設(shè)置聚類數(shù)目)
輸出:聚類結(jié)果
1.將樣本集中的所有的樣本點(diǎn)都當(dāng)做一個(gè)獨(dú)立的類簇;
repeat:
2.計(jì)算兩兩類簇之間的距離(后邊會(huì)做介紹),找到距離最小的兩個(gè)類簇c1和c2;
3.合并類簇c1和c2為一個(gè)類簇;
util: 達(dá)到聚類的數(shù)目或者達(dá)到設(shè)定的條件
例圖:
歐式距離的計(jì)算公式
類簇間距離的計(jì)算方法有許多種:
(1)就是取兩個(gè)類中距離最近的兩個(gè)樣本的距離作為這兩個(gè)集合的距離,也就是說(shuō),最近兩個(gè)樣本之間的距離越小,這兩個(gè)類之間的相似度就越大
(2)取兩個(gè)集合中距離最遠(yuǎn)的兩個(gè)點(diǎn)的距離作為兩個(gè)集合的距離
(3)把兩個(gè)集合中的點(diǎn)兩兩的距離全部放在一起求一個(gè)平均值,相對(duì)也能得到合適一點(diǎn)的結(jié)果。
e.g.下面是計(jì)算組合數(shù)據(jù)點(diǎn)(A,F)到(B,C)的距離,這里分別計(jì)算了(A,F)和(B,C)兩兩間距離的均值。
(4)取兩兩距離的中值,與取均值相比更加能夠解除個(gè)別偏離樣本對(duì)結(jié)果的干擾。
(5)求每個(gè)集合的中心點(diǎn)(就是將集合中的所有元素的對(duì)應(yīng)維度相加然后再除以元素個(gè)數(shù)得到的一個(gè)向量),然后用中心點(diǎn)代替集合再去就集合間的距離
實(shí)現(xiàn)
接下來(lái)以世界銀行樣本數(shù)據(jù)集進(jìn)行簡(jiǎn)單實(shí)現(xiàn)。該數(shù)據(jù)集以標(biāo)準(zhǔn)格式存儲(chǔ)在名為WBClust2013.csv的CSV格式的文件中。其有80行數(shù)據(jù)和14個(gè)變量。數(shù)據(jù)來(lái)源
為了使得結(jié)果可視化更加方便,我將最后一欄人口數(shù)據(jù)刪除了。并且在實(shí)現(xiàn)層次聚類之后加入PCA降維與原始結(jié)果進(jìn)行對(duì)比。
from scipy.cluster.hierarchy import linkage, dendrogram, fcluster import matplotlib.pyplot as plt import pandas as pd import numpy as np data = pd.read_csv('data/WBClust2013.csv') data.pop('Pop') # data.pop('RuralWater') # data.pop('CellPhone') # data.pop('LifeExp') data = data[:20] country = list(data['Country']) data.pop('Country') # 以下代碼為僅使用層次聚類 plt.figure(figsize=(9, 7)) plt.title("original data") mergings = linkage(data, method='average') # print(mergings) dendrogram(mergings, labels=country, leaf_rotation=45, leaf_font_size=8) plt.show() Z = linkage(data, method='average') print(Z) cluster_assignments = fcluster(Z, t=3.0, criterion='maxclust') print(cluster_assignments) for i in range(1, 4): print('cluster', i, ':') num = 1 for index, value in enumerate(cluster_assignments): if value == i: if num % 5 == 0: print() num += 1 print(country[index], end=' ') print() # 以下代碼為加入PCA進(jìn)行對(duì)比 class myPCA(): def __init__(self, X, d=2): self.X = X self.d = d def mean_center(self, data): """ 去中心化 :param data: data sets :return: """ n, m = data.shape for i in range(m): aver = np.sum(self.X[:, i])/n x = np.tile(aver, (1, n)) self.X[:, i] = self.X[:, i]-x def runPCA(self): # 計(jì)算協(xié)方差矩陣,得到特征值,特征向量 S = np.dot(self.X.T, self.X) S_val, S_victors = np.linalg.eig(S) index = np.argsort(-S_val)[0:self.d] Y = S_victors[:, index] # 得到輸出樣本集 Y = np.dot(self.X, Y) return Y # data_for_pca = np.array(data) # pcaObject=myPCA(data_for_pca,d=2) # pcaObject.mean_center(data_for_pca) # res=pcaObject.runPCA() # plt.figure(figsize=(9, 7)) # plt.title("after pca") # mergings = linkage(res,method='average') # print(mergings) # dendrogram(mergings,labels=country,leaf_rotation=45,leaf_font_size=8) # plt.show() # Z = linkage(res, method='average') # print(Z) # cluster_assignments = fcluster(Z, t=3.0, criterion='maxclust') # print(cluster_assignments) # for i in range(1,4): # print('cluster', i, ':') # num = 1 # for index, value in enumerate(cluster_assignments): # if value == i: # if num % 5 ==0: # print() # num+=1 # print(country[index],end=' ') # print()
兩次分類結(jié)果都是一樣的:
cluster 1 : China United States Indonesia Brazil Russian Federation Japan Mexico Philippines Vietnam Egypt, Arab Rep. Germany Turkey Thailand France United Kingdom cluster 2 : India Pakistan Nigeria Bangladesh cluster 3 : Ethiopia
通過(guò)樹(shù)狀圖對(duì)結(jié)果進(jìn)行可視化
原始樹(shù)狀圖:
PCA降維后的結(jié)果:
到此這篇關(guān)于python實(shí)現(xiàn)層次聚類的文章就介紹到這了,更多相關(guān)python層次聚類內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
python進(jìn)程和線程用法知識(shí)點(diǎn)總結(jié)
在本篇文章里小編給大家整理了關(guān)于python進(jìn)程和線程用法以及相關(guān)實(shí)例內(nèi)容,需要的朋友們跟著學(xué)習(xí)下。2019-05-05使用pygame實(shí)現(xiàn)垃圾分類小游戲功能(已獲校級(jí)二等獎(jiǎng))
這篇文章主要介紹了使用pygame實(shí)現(xiàn)垃圾分類小游戲功能(已獲校級(jí)二等獎(jiǎng)),本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-07-07python3?http.client?網(wǎng)絡(luò)請(qǐng)求方式
這篇文章主要介紹了python3?http.client?網(wǎng)絡(luò)請(qǐng)求方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-09-09python 參數(shù)列表中的self 顯式不等于冗余
Self in the Argument List: Redundant is not Explicit2008-12-12Python的selenium模塊使用find_element_by_id無(wú)效解決方案
這篇文章主要介紹了Python的selenium模塊使用find_element_by_id無(wú)效解決方案,find_element_by_id無(wú)效可能是因?yàn)榘姹締?wèn)題,而4.5.0版本不支持頁(yè)面對(duì)象的定位find_element_by_id方法,以前版本支持這些進(jìn)行元素定位,需要的朋友可以參考下2023-12-12解決matplotlib庫(kù)show()方法不顯示圖片的問(wèn)題
今天小編就為大家分享一篇解決matplotlib庫(kù)show()方法不顯示圖片的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-05-05Python3使用TCP編寫一個(gè)簡(jiǎn)易的文件下載器功能
這篇文章主要介紹了Python3使用TCP編寫一個(gè)簡(jiǎn)易的文件下載器功能,非常不錯(cuò),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2019-05-05pyinstaller參數(shù)介紹以及總結(jié)詳解
這篇文章主要介紹了pyinstaller參數(shù)介紹以及總結(jié)詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-07-07