赞
踩
K-means是一种经典的无监督学习算法,用于对数据进行聚类。K-means算法将数据集视为具有n个特征的n维空间,并尝试通过最小化簇内平方误差的总和来将数据点划分为簇。本文将介绍K-means算法的原理、实现和应用。
K-means算法是一种迭代算法,其基本思想是通过将每个数据点分配到最近的质心,并计算新的质心来迭代地改进簇的质量,直到质心不再变化或达到最大迭代次数为止。具体步骤如下:
K-means算法的核心是将数据点分配到最近的质心所在的簇,这是通过计算每个数据点与K个质心的距离来实现的。一般而言,距离可以使用欧氏距离、曼哈顿距离等来计算。而每个簇的质心则是该簇内所有数据点的均值,用于表示该簇的中心位置。
在K-means算法中,簇的数量k是需要事先指定的。选择合适的簇的数量非常重要。
需要注意的是,K-means算法可能陷入局部最优解,因此,选择k值需要多次运行算法,比较不同的聚类结果。
import numpy as np import matplotlib.pyplot as plt # 创建数据集 np.random.seed(0) n_samples = 1000 centers = np.array([[0, 0], [3, 3], [0, 3], [3, 0]]) X = np.zeros((n_samples, 2)) for i in range(len(centers)): X[i * (n_samples // len(centers)): (i + 1) * (n_samples // len(centers)), :] = \ centers[i] + np.random.randn(n_samples // len(centers), 2) # 可视化数据集 plt.scatter(X[:, 0], X[:, 1], s=10) plt.show()
import numpy as np def k_means(X, k, max_iter=100): # 随机选择k个点作为初始质心 centroids = X[np.random.choice(len(X), k, replace=False)] for i in range(max_iter): # 计算每个数据点与k个质心的距离 distances = np.linalg.norm(X[:, np.newaxis, :] - centroids, axis=-1) # 将数据点划分到距离最近的质心所在的簇 labels = np.argmin(distances, axis=1) # 对于每个簇,重新计算该簇内所有数据点的均值,将该均值作为新的质心 new_centroids = np.array([X[labels == j].mean(axis=0) for j in range(k)]) # 如果质心没有变化,则停止迭代 if np.allclose(new_centroids, centroids): break centroids = new_centroids return labels, centroids if __name__ == '__main__': labels, centroids = k_means(X=X, k=4) plt.scatter(X[labels==0, 0], X[labels==0, 1], color='r') plt.scatter(X[labels==1, 0], X[labels==1, 1], color='g') plt.scatter(X[labels==2, 0], X[labels==2, 1], color='b') plt.scatter(X[labels==3, 0], X[labels==3, 1]) plt.show()
import numpy as np import matplotlib.pyplot as plt # 创建数据集 np.random.seed(0) n_samples = 1000 centers = np.array([[0, 0], [3, 3], [0, 3], [3, 0]]) X = np.zeros((n_samples, 2)) for i in range(len(centers)): X[i * (n_samples // len(centers)): (i + 1) * (n_samples // len(centers)), :] = \ centers[i] + np.random.randn(n_samples // len(centers), 2) # 可视化数据集 def plot_data(X, labels): plt.scatter(X[labels==0, 0], X[labels==0, 1], color='r') plt.scatter(X[labels==1, 0], X[labels==1, 1], color='g') plt.scatter(X[labels==2, 0], X[labels==2, 1], color='b') plt.scatter(X[labels==3, 0], X[labels==3, 1], color='m') plt.show() def plot_centers(centroids): plt.scatter(centroids[:, 0], centroids[:, 1], s=200, marker='*', color='k') plt.show() def k_means(X, k, max_iter=100): # 随机选择k个点作为初始质心 centroids = X[np.random.choice(len(X), k, replace=False)] for i in range(max_iter): # 计算每个数据点与k个质心的距离 distances = np.linalg.norm(X[:, np.newaxis, :] - centroids, axis=-1) # 将数据点划分到距离最近的质心所在的簇 labels = np.argmin(distances, axis=1) # 对于每个簇,重新计算该簇内所有数据点的均值,将该均值作为新的质心 new_centroids = np.array([X[labels == j].mean(axis=0) for j in range(k)]) # 如果质心没有变化,则停止迭代 if np.allclose(new_centroids, centroids): break centroids = new_centroids return labels, centroids if __name__ == '__main__': labels, centroids = k_means(X=X, k=4) plot_data(X, labels) plot_centers(centroids)
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。