聚类算法及可视化方法的实践与探索

簇内平方和表示数据点到其簇内质心的距离的平方和,公式如下:

其中, 是k簇数, ni是第 i 个簇的样本数, xij是第  i个簇中的第 j 个样本。

import matplotlib.pyplot as plt
from sklearn.cluster import KMeans
from sklearn.datasets import make_blobs

# 生成模拟数据
X, _ = make_blobs(n_samples=300, centers=4, random_state=42)

# 尝试不同的簇数,计算簇内平方和
inertia = []
for k in range(1, 11):
    kmeans = KMeans(n_clusters=k, random_state=42)
    kmeans.fit(X)
    inertia.append(kmeans.inertia_)

# 绘制肘部法图像
plt.plot(range(1, 11), inertia, marker='o')
plt.xlabel('Number of Clusters (K)')
plt.ylabel('Inertia')
plt.title('Elbow Method for Optimal K')
plt.show()

在图像中,我们寻找一个肘部,即簇内平方和的变化趋势减缓的点。这个肘部对应的簇数就是我们的最佳选择。

需要注意,有时候肘部并不明显,这时可能需要结合业务背景和其他评估指标来综合判断最佳的簇数。

图片

层次聚类的链接方式

层次聚类的链接方式主要有单链接(single-linkage)和完全链接(complete-linkage)两种,它们在形成聚类时有着不同的原理和效果。

1. 单链接 (Single-linkage):

在单链接中,两个聚类的距离被定义为它们中最近的两个点之间的距离。

也就是说,如果一个簇中的某个点与另一个簇中的所有点的距离都很小,那么这两个簇将会很接近。

2. 完全链接 (Complete-linkage):

在完全链接中,两个聚类的距离被定义为它们中最远的两个点之间的距离。这意味着两个簇被认为是在它们中最远的点之间保持一致的情况下被连接。

选择方式:

  • 如果你的数据可能包含离群点,并且你希望对异常值具有一定的鲁棒性,可以选择完全链接。

  • 如果你的数据相对干净,且你希望得到更细致的簇,可以选择单链接。

import numpy as np
from scipy.spatial.distance import pdist, squareform
from scipy.cluster.hierarchy import linkage, dendrogram
import matplotlib.pyplot as plt

# 生成一些示例数据
np.random.seed(42)
data = np.random.rand(10, 2)

# 计算距离矩阵
distance_matrix = squareform(pdist(data))

# 使用单链接层次聚类
single_linkage = linkage(distance_matrix, method='single')

# 使用完全链接层次聚类
complete_linkage = linkage(distance_matrix, method='complete')

# 绘制树状图
plt.figure(figsize=(12, 6))

plt.subplot(1, 2, 1)
dendrogram(single_linkage)
plt.title('Single Linkage')

plt.subplot(1, 2, 2)
dendrogram(complete_linkage)
plt.title('Complete Linkage')

plt.show()

这段代码使用了scipy库中的linkagedendrogram函数来进行单链接和完全链接的层次聚类,并绘制了树状图,有助于理解聚类的形成过程。

图片

  1. 了解数据特性: 首先,你需要了解你的数据集特性,包括样本分布、聚类的密度和大小。不同的数据集可能需要不同的参数设置。

  2. 可视化数据: 使用散点图等可视化工具来观察数据的分布,这可以帮助你判断合适的邻域半径和最小样本数。可视化工具可以是matplotlib库,用于绘制数据分布图。

  3. 尝试不同参数组合: 通过尝试不同的邻域半径和最小样本数的组合,观察聚类结果的变化。通常,可以通过在一定范围内进行网格搜索来找到最优的参数组合。可以使用嵌套循环来遍历参数空间,然后评估每种组合的聚类效果。

  4. 使用密度可视化: 利用DBSCAN的核心思想是根据密度连接点,你可以通过在邻域内绘制样本点的个数来进行密度可视化。这有助于理解在不同参数下的数据密度分布。

下面用PyTorch实现DBSCAN的代码,并用matplotlib画出密度可视化图:

import torch
from sklearn.datasets import make_blobs
import matplotlib.pyplot as plt
from sklearn.cluster import DBSCAN

# 生成一些示例数据
X, _ = make_blobs(n_samples=300, centers=3, cluster_std=0.60, random_state=0)

# 转换为PyTorch张量
X_tensor = torch.tensor(X, dtype=torch.float32)

# DBSCAN算法
dbscan = DBSCAN(eps=0.3, min_samples=5)
labels = dbscan.fit_predict(X_tensor)

# 绘制密度可视化图
plt.scatter(X[:, 0], X[:, 1], c=labels, cmap='viridis', s=50, edgecolors='k')
plt.title('DBSCAN Clustering')
plt.xlabel('Feature 1')
plt.ylabel('Feature 2')
plt.show()

通过调整epsmin_samples参数,你可以观察到不同的聚类效果。密度可视化图能够帮助你更直观地理解不同参数组合下的聚类结果。

图片

聚类算法的时间复杂度

Mini-Batch K-Means:

K-Means是一种常见的聚类算法,但在大规模数据上运行时可能效率较低。Mini-Batch K-Means是对传统K-Means的一种改进,通过每次随机选取一小部分数据来更新簇中心,从而减少计算量。

可以通过以下步骤实现:

 from sklearn.cluster import MiniBatchKMeans
 kmeans = MiniBatchKMeans(n_clusters=3, batch_size=100)
 kmeans.fit(X)

使用近似算法:

对于大规模数据,可以考虑使用一些近似聚类算法,例如K-Means++、K-Means||。这些算法在初始簇中心的选择上更为高效,有助于加速整个聚类过程。

谱聚类:

谱聚类是一种基于图论的聚类方法,可以通过降维来减少计算复杂度。它将数据转换到低维空间,然后在该空间中执行K-Means聚类。在大规模数据集上,通过适当的降维,可以提高计算效率。

 from sklearn.cluster import SpectralClustering
 spectral = SpectralClustering(n_clusters=3, affinity='nearest_neighbors')
 spectral.fit(X)

使用分布式计算:

对于大规模数据,分布式计算是一种有效的方式。可以考虑使用分布式聚类算法,例如Spark中的K-Means。

K-Means并行化:

通过使用多核或GPU进行并行化,可以加速K-Means的计算过程。在Python中,可以使用joblib库来实现多核并行计算。

 from sklearn.cluster import KMeans
 from joblib import parallel_backend
 with parallel_backend('threading', n_jobs=2):
     kmeans = KMeans(n_clusters=3)
     kmeans.fit(X)

总体而言,在大规模数据上提高聚类算法的计算效率,关键是选择合适的算法、采用近似方法、并行化计算,以及利用分布式计算资源。根据具体问题的性质和数据特点,选择最合适的策略来优化算法的性能。

聚类结果的可解释性

如何更好地解释这些簇对应的数据特征呢

1. 可视化聚类结果: 使用降维技术(如t-SNE、PCA)将高维数据映射到二维或三维空间,然后用不同颜色或标记表示不同的簇。这有助于直观地观察数据点在聚类中的分布。

# 以t-SNE为例
from sklearn.manifold import TSNE
import matplotlib.pyplot as plt

tsne = TSNE(n_components=2, random_state=42)
clustered_data_tsne = tsne.fit_transform(clustered_data)

plt.scatter(clustered_data_tsne[:, 0], clustered_data_tsne[:, 1], c=labels, cmap='viridis')
plt.title('t-SNE Visualization of Clusters')
plt.show()

2. 簇内数据统计: 对每个簇进行统计分析,计算每个簇的中心(均值)、方差等统计量。

可以通过pandas库来实现。

import pandas as pd

# 假设data是你的原始数据,labels是聚类结果
clustered_data = pd.DataFrame(data, columns=['feature1', 'feature2', ...])
clustered_data['label'] = labels

# 统计每个簇的均值
cluster_means = clustered_data.groupby('label').mean()
print(cluster_means)

3. 特征重要性分析: 对每个簇内数据的特征进行重要性分析,可以使用决策树等模型。这能够帮助你理解哪些特征对于簇的形成起到了重要作用。

from sklearn.tree import DecisionTreeClassifier

# 假设X是你的原始数据,y是聚类结果
dt = DecisionTreeClassifier(random_state=42)
dt.fit(X, y)

# 特征重要性
feature_importance = dt.feature_importances_
print("Feature Importance:", feature_importance)

4. 关联规则挖掘: 使用关联规则挖掘方法,如Apriori算法,挖掘在同一簇内经常出现的特征组合。

from mlxtend.frequent_patterns import apriori
from mlxtend.frequent_patterns import association_rules

# 假设X是你的原始数据,y是聚类结果
frequent_itemsets = apriori(X, min_support=0.5, use_colnames=True)
rules = association_rules(frequent_itemsets, metric="confidence", min_threshold=0.7)
print(rules)

5. 使用主成分分析(PCA): 对每个簇内的数据应用PCA,找到主成分,以更好地理解簇的结构。

from sklearn.decomposition import PCA

# 假设clustered_data是你的聚类结果
pca = PCA(n_components=2)
clustered_data_pca = pca.fit_transform(clustered_data)

plt.scatter(clustered_data_pca[:, 0], clustered_data_pca[:, 1], c=labels, cmap='viridis')
plt.title('PCA Visualization of Clusters')
plt.show()

以上方法可以帮助你更好地解释聚类结果,理解每个簇对应的数据特征,进而为进一步分析和应用提供更多的线索。

处理高维数据

下面是一些处理高维数据的方法:

1. 降维技术:

  • 主成分分析(PCA): 通过线性变换将原始高维数据投影到低维空间,保留最大方差的方向,以减少数据的维度。

  • t分布邻域嵌入(t-SNE): 一种非线性降维方法,能够在保持样本之间的相对距离的同时降低维度。

# 使用PCA进行降维
from sklearn.decomposition import PCA

# 假设X是高维数据
pca = PCA(n_components=2)
X_low_dim = pca.fit_transform(X)

2. 特征选择:

  • 通过选择最相关的特征,剔除冗余信息,从而降低数据的维度。

# 使用方差选择法进行特征选择
from sklearn.feature_selection import VarianceThreshold

selector = VarianceThreshold(threshold=0.5)
X_low_dim = selector.fit_transform(X)

3. 使用聚类前的预处理:

  • 在应用聚类算法之前,可以先对数据进行预处理,如归一化、标准化,以确保每个维度的重要性相当。

# 使用MinMaxScaler进行归一化
from sklearn.preprocessing import MinMaxScaler

scaler = MinMaxScaler()
X_normalized = scaler.fit_transform(X)

4. 密度聚类算法:

  • 与基于距离的聚类算法不同,密度聚类算法(如DBSCAN)可以在高维空间中更灵活地发现密集的数据点,而不受维度的影响。

# 使用DBSCAN进行密度聚类
from sklearn.cluster import DBSCAN

dbscan = DBSCAN(eps=0.5, min_samples=5)
labels = dbscan.fit_predict(X)

5. 集成聚类算法:

  • 使用多个聚类算法并集成它们的结果,可以提高在高维数据上的聚类效果。

# 使用KMeans和AgglomerativeClustering进行集成聚类
from sklearn.cluster import KMeans, AgglomerativeClustering
from sklearn.ensemble import VotingClassifier

kmeans = KMeans(n_clusters=3)
agg_clustering = AgglomerativeClustering(n_clusters=3)

ensemble_clf = VotingClassifier(estimators=[('kmeans', kmeans), ('agg', agg_clustering)], voting='hard')
labels = ensemble_clf.fit_predict(X)

6. 可视化:

  • 对降维后的数据进行可视化,以便更好地理解数据的结构和聚类效果。

# 使用matplotlib进行可视化
import matplotlib.pyplot as plt

plt.scatter(X_low_dim[:, 0], X_low_dim[:, 1], c=labels, cmap='viridis')
plt.title('Clustering Result in 2D Space')
plt.show()

通过采用这些方法,我们可以更好地处理高维数据上的聚类问题,提高算法的性能和可解释性。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/259422.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Python Opencv实践 - 手势音量控制

本文基于前面的手部跟踪功能做一个手势音量控制功能,代码用到了前面手部跟踪封装的HandDetector.这篇文章在这里: Python Opencv实践 - 手部跟踪-CSDN博客文章浏览阅读626次,点赞11次,收藏7次。使用mediapipe库做手部的实时跟踪&…

47 星南二楼

动态规划&#xff0c;相当于求解最长子序列问题 #include <iostream> using namespace::std; using std::cout; using std::cin; int n; int a[5100],dp[5100];int xnel(int n, int a[]) {int result 0;for(int i0; i<n; i){for(int j0; j<i; j){if(a[j]<a[…

node版本与npm版本不对应的解决方案

大家好&#xff0c;我是雄雄&#xff0c;欢迎关注微信公众号&#xff1a;雄雄的小课堂 错误情况 今天遇到了个问题&#xff0c;原来用的node的版本是v14.16.0的&#xff0c;后来升级到了最新版20多的时候&#xff0c;以前的项目启动不了。 于是我手动将node卸载了&#xff0c…

【2023年网络安全优秀创新成果大赛专刊】银行数据安全解决方案(天空卫士)

在2023年网络安全优秀创新成果大赛&#xff0c;成都分站中&#xff0c;天空卫士银行数据安全方案获得优秀解决方案奖。与此同时&#xff0c;天空卫士受信息安全杂志邀请&#xff0c;编写《银行数据安全解决方案》。12月6日&#xff0c;天空卫士编写的《银行数据安全解决方案》做…

DAY11

问题一&#xff1a;指针与引用的区别 疑问 为什么引用的本质是指针常量&#xff0c;但是对它求sizeof却是变量所占内存空间的大小那&#xff1f;&#xff1f;&#xff1f;&#xff1f;&#xff1f;&#xff1f;&#xff1f;&#xff1f;&#xff1f;&#xff1f; 1.引用是给变…

Playground AI刚刚推出了它的新宠儿——Playground V2,去试试?

每周跟踪AI热点新闻动向和震撼发展 想要探索生成式人工智能的前沿进展吗&#xff1f;订阅我们的简报&#xff0c;深入解析最新的技术突破、实际应用案例和未来的趋势。与全球数同行一同&#xff0c;从行业内部的深度分析和实用指南中受益。不要错过这个机会&#xff0c;成为AI领…

令人惊叹的代码技巧

在编程世界中&#xff0c;有一些令人惊叹的代码技巧和巧妙的实现方式。以下是一些我见过的令人印象深刻的代码技巧&#xff1a; 函数式编程魔法&#xff1a; 使用函数式编程的一些特性&#xff0c;比如高阶函数、匿名函数和Lambda表达式&#xff0c;可以使代码更为简洁、易读。…

(9)Linux Git的介绍以及缓冲区

&#x1f4ad; 前言 本章我们先对缓冲区的概念进行一个详细的探究&#xff0c;之后会带着大家一步步去编写一个简陋的 "进度条" 小程序。最后我们来介绍一下 Git&#xff0c;着重讲解一下 Git 三板斧&#xff0c;一般只要掌握三板斧就基本够用了。 缓冲区&#xff…

Redis权限管理体系(三): ACL 配置持久化

点击上方蓝字关注我 前面我们已经了解了ACL用户管理的用途及使用&#xff1a; Redis权限管理体系(一&#xff09;&#xff1a;客户端名及用户名 Redis权限管理体系(二&#xff09;&#xff1a;终于等来了Redis权限控制体系ACL 但因默认配置中ACL的配置未持久化&#xff0c;因此…

oracle怎样才算开启了内存大页?

oracle怎样才算开启了内存大页&#xff1f; 关键核查下面三点&#xff1a; 1./etc/sysctl.conf vm.nr_hugepages16384这是给了32G&#xff0c;计划sga给30G&#xff0c;一般需多分配2-4G sysctl -p生效 看cat /proc/meminfo|grep Huge啥结果&#xff1f; 这种明显是配了…

Go项目快速集成Swagger UI

swag Swag将Go的注释转换为Swagger2.0文档。我们为流行的 Go Web Framework 创建了各种插件&#xff0c;这样可以与现有Go项目快速集成&#xff08;使用Swagger UI&#xff09;。 目录 快速开始支持的Web框架如何与Gin集成格式化说明开发现状声明式注释格式 通用API信息API操…

【每日一题】【12.20】2828.判别首字母缩略词

&#x1f525;博客主页&#xff1a; A_SHOWY&#x1f3a5;系列专栏&#xff1a;力扣刷题总结录 数据结构 云计算 数字图像处理 力扣每日一题_ 1.题目链接 2828. 判别首字母缩略词https://leetcode.cn/problems/check-if-a-string-is-an-acronym-of-words/ 2.题目描述 今天…

RTDETR论文快速理解和代码快速实现(训练与预测)

文章目录 前言一、摘要二、论文目的三、论文贡献四、模型结构1、模型整体结构2、backbone结构3、neck结构4、混合编码器(neck) 五、RTDERT模型训练(data-->train)1、环境安装2、训练1、数据准备2、数据yaml文件3、训练代码4、训练运行结果 3、推理1、推理代码2、推理运行结果…

Nginx快速入门:安装目录结构详解及核心配置解读(二)

0. 引言 上节我们讲解了nginx的应用场景和安装&#xff0c;本节继续针对nginx的各个目录文件进行讲解&#xff0c;让大家更加深入的认识nginx。并通过一个实操案例&#xff0c;带大家来实际认知nginx的核心配置 1. nginx安装目录结构 首先nginx的默认安装目录为&#xff1a;…

原子学习笔记3——使用tslib库

一、tslib介绍 tslib 是专门为触摸屏设备所开发的 Linux 应用层函数库&#xff0c;并且是开源。 tslib 为触摸屏驱动和应用层之间的适配层&#xff0c;它把应用程序中读取触摸屏 struct input_event 类型数据&#xff08;这是输入设备上报给应用层的原始数据&#xff09;并进行…

Elasticsearch常见面试题

文章目录 1.简单介绍下ES&#xff1f;2.简单介绍当前可以下载的ES稳定版本&#xff1f;3.安装ES前需要安装哪种软件&#xff1f;4.请介绍启动ES服务的步骤&#xff1f;5.ES中的倒排索引是什么&#xff1f;6. ES是如何实现master选举的&#xff1f;7. 如何解决ES集群的脑裂问题8…

SpringBoot-XXLJOB提供动态API调度任务

目录 一、项目版本 二、XXL-JOB提供动态API controller层 service层 三、SpringBoot项目 pom model XxlJobUtil-工具类 XXL-JOB是一个分布式任务调度平台&#xff0c;其核心设计目标是开发迅速、学习简单、轻量级、易扩展。现已开放源代码并接入多家公司线上产品线&…

Gateway网关-网关的cors跨域配置

目录 一、跨域 二、解决方案 三、实际测试 3.1 html调用接口 3.2 跨域问题复现 3.3 application文件中配置CORS 3.4 问题解决 一、跨域 跨域问题&#xff1a;浏览器禁止请求的发起者与服务端发生跨域ajax请求&#xff0c;请求被浏览器拦截的问题 跨域&#xff1a;…

第四节TypeScript 声明变量

1、typescript变量声明 变量是一种使用方便的占位符&#xff0c;用于引用计算机内存地址。 我们可以把变量看做存储数据的容器。 typescript变量的命名规则&#xff1a; 变量名称可以包含数字和字母。除了下划线_和美元$符号外&#xff0c;不能包含其它特殊字符&#xff0c…

基于Alpha-Beta剪枝树的井字棋人机博弈系统的实现

这篇文章讨论了算法的基本概念与特性&#xff0c;并介绍了五种常见的算法类型&#xff1a;分治法、动态规划、贪心算法、回溯法和分支限界法。文章以井字棋博弈中的Alpha-Beta剪枝树作为示例&#xff0c;详细解释了该算法的应用和原理。Alpha-Beta剪枝树是一种用于实现游戏AI的…