当前位置: 首页 > news >正文

DBSCAN 算法【python,机器学习,算法】

DBSCAN 即 Density of Based Spatial Clustering of Applications with Noise,带噪声的基于空间密度聚类算法。

算法步骤:

  1. 初始化:
    • 首先,为每个数据点分配一个初始聚类标签,这里设为0,表示该点尚未被分配到一个聚类中。
    • 设置一个聚类ID(cluster_id),初始化为0,用于标识不同的聚类。
  2. 遍历数据点:
    遍历数据集中的每个点。如果某点已经被标记(即不属于聚类0),则跳过该点。
  3. 查找邻居点:
    对于每个尚未被标记的点,使用get_neighbors函数查找其ε-邻域内的所有邻居点。这通常是通过计算该点与数据集中其他点之间的欧氏距离,并比较距离与ε来实现的。
  4. 处理邻居点数量:
    • 如果找到的邻居点数量小于min_pts(最小邻居数量),则将当前点标记为噪声点(标签设为-1)。
    • 如果邻居点数量大于或等于min_pts,则将该点标记为一个新的聚类(将cluster_id加1,并将该点标签设为新的cluster_id)。
  5. 扩展聚类:
    • 对于每个新发现的聚类中的点(即刚被标记为当前cluster_id的点),执行expand_cluster函数以进一步扩展聚类。
    • 在expand_cluster函数中,遍历当前点的所有邻居点,并根据其标签进行处理:
      • 如果邻居点是噪声点(标签为-1),则将其标记为当前聚类(将标签改为cluster_id)。
      • 如果邻居点尚未被标记(标签为0),则将其标记为当前聚类,并递归地查找并标记其邻居点(如果其邻居点数量也满足min_pts)。
  6. 返回结果:
    当所有点都被处理完毕后,算法返回每个数据点的最终聚类标签。

下面是代码实现:

from collections import Counterimport numpy as np
from sklearn.datasets import make_blobsdef dbscan(data, eps, min_pts):# 初始化每个数据点的聚类标签为 0labels = [0] * len(data)# 聚类 idcluster_id = 0for i in range(len(data)):if labels[i] != 0:# 如果数据点已经被标记过,则跳过该点,继续下一个点continue# 获取当前点的邻居点neighbors = get_neighbors(data, i, eps)# 如果邻居点的数量小于最小邻居数量,则将当前点标记为噪声点if len(neighbors) < min_pts:labels[i] = -1else:# 否则,增加聚类 idcluster_id += 1# 将当前点标记为当前聚类 idlabels[i] = cluster_id# 扩展聚类expand_cluster(data, labels, neighbors, cluster_id, eps, min_pts)# 返回每个数据点的聚类标签return labelsdef expand_cluster(data, labels, neighbors, cluster_id, eps, min_pts):# 遍历每个邻居点for neighbor in neighbors:# 如果邻居点的标签为 -1if labels[neighbor] == -1:# 将噪声点标记为当前聚类 idlabels[neighbor] = cluster_id# 如果邻居点的标签为 0elif labels[neighbor] == 0:# 将邻居点标记为当前聚类 idlabels[neighbor] = cluster_id# 获取邻居点的邻居点new_neighbors = get_neighbors(data, neighbor, eps)# 如果新的邻居点数量满足最小邻居数量要求,则将其加入邻居列表if len(new_neighbors) >= min_pts:neighbors += new_neighborsdef get_neighbors(data, point_idx, eps):# 邻居点列表neighbors = []for i in range(len(data)):# 计算当前点与目标点之间的欧氏距离,如果距离小于邻域半径 epsif np.linalg.norm(data[i] - data[point_idx]) < eps:# 将目标点的索引加入邻居点列表neighbors.append(i)# 返回邻居点列表return neighborsnp.random.seed(0)
# 生成样例数据
data, y = make_blobs(n_samples=200, centers=5, cluster_std=0.6)
print(Counter(y))eps, min_pts = 0.6, 3
# 进行聚类
labels = dbscan(data, eps, min_pts)
print(Counter(labels))

上述代码实现了一个简单的 DBSCAN 算法。注意,在实际应用中,你需要根据实际情况调整邻域半径参数和核心点周围最小数据点数。
一般情况下,最小数据点数取数据维度值的 2 倍数,最小取 3。 该参数越大,可能的噪声点会被聚类,同样的邻域半径越小,噪声点也会被分类。

相关文章:

  • Web前端职业描述:编织数字世界的绚丽画卷
  • 360数字安全:2024年4月勒索软件流行态势分析报告
  • MySQL常用命令(Linux环境)
  • 2024年,计算机相关专业还值得选择吗?
  • 如何实现观察者模式和发布-订阅模式?
  • 高压电工作业历年试题分享(含答案)
  • web鼠标前端设置:深入探索与个性化定制
  • 不是所有的硬盘柜都叫“安全专家”,但它做到了!
  • Lua - 魔兽世界SRP6网站源码(FastWeb)
  • Nginx05-负载均衡详解、LNMP+NFS、会话保持、负载均衡状态检查upstream-check、平滑升级
  • 大功率回馈式负载:行业竞争态势
  • 线性数据结构-栈
  • 【QEMU中文手册】2.2 调用方式(持续更新中)
  • Mimio安装
  • js实现简单计算器词法解析语法解析解释器,带可视化界面
  • [微信小程序] 使用ES6特性Class后出现编译异常
  • 《微软的软件测试之道》成书始末、出版宣告、补充致谢名单及相关信息
  • emacs初体验
  • iOS动画编程-View动画[ 1 ] 基础View动画
  • react-native 安卓真机环境搭建
  • ReactNativeweexDeviceOne对比
  • SpiderData 2019年2月16日 DApp数据排行榜
  • 对超线程几个不同角度的解释
  • 基于遗传算法的优化问题求解
  • 前端 CSS : 5# 纯 CSS 实现24小时超市
  • 如何胜任知名企业的商业数据分析师?
  • 入门级的git使用指北
  • 线性表及其算法(java实现)
  • 用Python写一份独特的元宵节祝福
  • 原生Ajax
  • 在 Chrome DevTools 中调试 JavaScript 入门
  • # Redis 入门到精通(九)-- 主从复制(1)
  • #我与Java虚拟机的故事#连载09:面试大厂逃不过的JVM
  • (2/2) 为了理解 UWP 的启动流程,我从零开始创建了一个 UWP 程序
  • (5)STL算法之复制
  • (7)STL算法之交换赋值
  • (附源码)spring boot网络空间安全实验教学示范中心网站 毕业设计 111454
  • (论文阅读笔记)Network planning with deep reinforcement learning
  • (原創) 人會胖會瘦,都是自我要求的結果 (日記)
  • *Algs4-1.5.25随机网格的倍率测试-(未读懂题)
  • .NET 4 并行(多核)“.NET研究”编程系列之二 从Task开始
  • .NET 项目中发送电子邮件异步处理和错误机制的解决方案
  • .Net6支持的操作系统版本(.net8已来,你还在用.netframework4.5吗)
  • .Net程序猿乐Android发展---(10)框架布局FrameLayout
  • .Net开发笔记(二十)创建一个需要授权的第三方组件
  • .NET开源项目介绍及资源推荐:数据持久层 (微软MVP写作)
  • .net项目IIS、VS 附加进程调试
  • ??myeclipse+tomcat
  • @manytomany 保存后数据被删除_[Windows] 数据恢复软件RStudio v8.14.179675 便携特别版...
  • @RequestParam @RequestBody @PathVariable 等参数绑定注解详解
  • @软考考生,这份软考高分攻略你须知道
  • [ C++ ] STL_list 使用及其模拟实现
  • [ vulhub漏洞复现篇 ] JBOSS AS 5.x/6.x反序列化远程代码执行漏洞CVE-2017-12149
  • []指针
  • [AR]Vumark(下一代条形码)