Scipy簇聚


K均值聚類是一種在一組未標記資料中查詢聚類和聚類中心的方法。 直覺上,我們可以將一個群集(簇聚)看作 - 包含一組資料點,其點間距離與群集外點的距離相比較小。 給定一個K中心的初始集合,K均值演算法重複以下兩個步驟 -

  • 對於每個中心,比其他中心更接近它的訓練點的子集(其聚類)被識別出來。
  • 計算每個聚類中資料點的每個要素的平均值,並且此平均向量將成為該聚類的新中心。

重複這兩個步驟,直到中心不再移動或分配不再改變。 然後,可以將新點x分配給最接近的原型的群集。 SciPy庫通過叢集包提供了K-Means演算法的良好實現。 下面來了解如何使用它。

SciPy中實現K-Means

我們來看看並理解如何在SciPy中實現K-Means。

匯入K-Means

下面來看看每個匯入的函式的實現和用法。

from SciPy.cluster.vq import kmeans,vq,whiten

資料生成

我們需要生成(模擬)一些資料來探索聚類。參考以下程式碼 -

from numpy import vstack,array
from numpy.random import rand

# data generation with three features
data = vstack((rand(100,3) + array([.5,.5,.5]),rand(100,3)))

現在,我們來看看生成的模擬資料,上述程式將生成以下輸出。

[[ 1.34103331  1.13924682  0.68465819]
 [ 1.28481332  0.91318917  0.84225546]
 [ 0.96498008  1.42382266  0.83564809]
 [ 1.37049373  0.66635033  1.46568707]
 [ 0.87424166  0.86090225  1.22545336]
 [ 1.0264795   0.90724604  1.46837972]
 [ 1.40996857  1.37769991  1.39805802]
 [ 0.964556    0.71632157  1.47983347]
 [ 0.69909637  1.21695335  1.46434369]
 [ 1.01887602  0.86448455  1.02242951]
 [ 0.82573176  1.19165063  1.09085707]
 [ 0.64378227  0.70673944  0.69484097]
 [ 1.16087103  0.64371977  0.89720984]
 [ 1.23410673  0.56805382  1.33534058]
 [ 0.50417695  1.29632466  0.96589447]
 [ 0.91395183  1.39173555  1.0748435 ]
 [ 1.04540644  1.20721464  0.97173727]
 ... ...
 [ 0.79250839  0.48689797  0.42250824]
 [ 0.05846914  0.83469742  0.57586067]
 [ 0.0308333   0.8642561   0.1111777 ]
 [ 0.61327069  0.43425013  0.99716439]
 [ 0.81698148  0.91098877  0.12706862]
 [ 0.60665992  0.55999208  0.57454962]
 [ 0.13894142  0.03315365  0.43182983]
 [ 0.62293781  0.34701877  0.61229591]]

根據每個要素標準化一組觀察值。 在執行K-Means之前,使用白化重新縮放觀察集的每個特徵維度是有好處的。 每個特徵除以所有觀測值的標準偏差以給出其單位差異。

美化資料

我們可使用以下程式碼來美白資料。

# whitening of data
data = whiten(data)
print (data)

用三個叢集計算K均值

現在使用以下程式碼計算三個群集的K均值。

# computing K-Means with K = 3 (2 clusters)
centroids,_ = kmeans(data,3)

上述程式碼對形成K個簇的一組觀測向量執行K均值。 K-Means演算法調整質心直到不能獲得足夠的進展,即失真的變化,因為最後一次疊代小於某個閾值。 在這裡,可以通過使用下面給出的程式碼列印centroids變數來觀察簇。

print(centroids)

上面的程式碼將生成以下輸出。

print(centroids)[ [ 2.26034702  1.43924335  1.3697022 ]
                  [ 2.63788572  2.81446462  2.85163854]
                  [ 0.73507256  1.30801855  1.44477558] ]

使用下面給出的程式碼將每個值分配給一個叢集。

# assign each sample to a cluster
clx,_ = vq(data,centroids)

vq函式將'M'中的每個觀察向量與'N' obs陣列與centroids進行比較,並將觀察值分配給最近的聚類。 它返回每個觀察和失真的聚類。 我們也可以檢查失真。使用下面的程式碼檢查每個觀察的聚類。

# check clusters of observation
print (clx)

上面的程式碼將生成以下輸出。

array([1, 1, 0, 1, 1, 1, 0, 1, 0, 0, 1, 1, 1, 0, 0, 1, 1, 0, 0, 1, 0, 2, 0, 2, 0, 1, 1, 1,
0, 0, 1, 1, 0, 0, 1, 1, 0, 0, 0, 0, 1, 1, 0, 0, 1, 1, 0, 0, 1, 0, 1, 0, 0, 0, 1, 1, 0, 0,
0, 1, 0, 1, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0,
0, 1,  0, 0, 0, 0, 1, 0, 0, 1, 2, 1, 2, 2, 2, 2, 2, 2, 2, 2, 0, 2, 0, 2, 2, 2, 2, 2, 0, 0,
2, 2, 2, 1, 0, 2, 0, 2, 2, 2, 2, 2, 2, 2, 2, 2, 0, 2, 2, 1, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 
2, 2, 0, 2, 0, 2, 2, 2, 2, 2, 2, 2, 2, 2, 0, 2, 2, 2, 2, 0, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
2, 2, 2, 2, 2, 2, 1, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2], dtype=int32)

上述陣列的不同值 - 0,1,2表示簇。