【数学与算法】KMeans聚类代码

2024-06-16 21:18

本文主要是介绍【数学与算法】KMeans聚类代码,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

KMeans聚类是根据各点距离聚类中心的距离来把所有点分类到不同类别的无监督算法。

对于聚类,就是两点:

  • 1.分类所有样本点:遍历每个数据样本点,分别计算该样本点与K个聚类中心的距离,把该样本点的类别重新分类为距离最小的那一类。
  • 2.更新聚类中心:所有样本点都按第一步重新分类后,把各类别的点重新计算聚类中心(求平均值的方法),更新K个类别的聚类中心值。
  • 3.重复前面两步,直到聚类中心点更新幅度小于阈值,或者达到迭代次数,或者所有样本点的类别都不再改变,或者他们几者组合起来,就停止迭代。

它适合分类一堆一堆的点:见下图中左边的三堆点。
不适合对几条曲线组成的点进行分类,见下图的右边三条线

以一条曲线的起点和终点为例:一条曲线特别长,他的起点和终点之间的距离可能也会特别大,因此,通过欧氏距离进行聚类的话,会出现别的曲线上的点更接近他的起点和终点的情况,那么起点和终点可能会和其他曲线很靠近的点聚类成一类。因此最终的分类效果肯定是很差。
所以,理解了聚类的原理,就知道了他的适用范围,不会在不能使用聚类的地方尝试使用聚类方法。

在这里插入图片描述


代码:

下面例子用kmeans分类一系列三维空间点。
头文件:

#pragma once
#include <cstring>
// #include <fstream>
#include <vector>struct Point_3D {float x;float y;float z;Point_3D operator=(Point_3D point) {x = point.x;y = point.y;z = point.z;}
};
typedef std::vector<Point_3D> Point3DVct;class KMeans {public:int m_k;  // k个类别Point3DVct input_point3D_vct_;          //要聚类的点云std::vector<Point3DVct> k_points_vct_;  // K类,每一类存储若干点Point3DVct k_center_point_vct_;         //每个类的中心KMeans() { m_k = 0; }inline void SetK(int k_) {m_k = k_;k_points_vct_.resize(m_k);}//设置输入点bool SetInput(const Point3DVct &input_points, Point3DVct &o_points);//初始化最初的K个类的中心bool InitKCenter(Point3DVct &K_center_point_vct);//聚类bool Cluster(const Point3DVct &input_points,std::vector<Point3DVct> &k_points_vct);//更新K类的中心bool UpdateGroupCenter(std::vector<Point3DVct> &K_points_vct,Point3DVct &centers);//计算两个点间的欧氏距离float DistBetweenPoints(const Point_3D &p1, const Point_3D &p2);//是否存在中心点移动,用来判断分类结果是否已收敛bool ExistCenterShift(Point3DVct &prev_center, Point3DVct &cur_center);
};

源文件:

#include "k_means.h"#include <math.h>
// #include <stdlib.h>
#include <bits/stdc++.h>
#include <time.h>#include <iostream>const float DELTA = 0.001;bool KMeans::InitKCenter(Point3DVct &K_center_point_vct) {if (m_k == 0) {std::cout << "在此之前必须要调用setK()函数" << std::endl;return false;}k_center_point_vct_.resize(m_k);for (size_t i = 0; i < m_k; ++i) {k_center_point_vct_[i] = K_center_point_vct[i];}return true;
}bool KMeans::SetInput(const Point3DVct &input_points, Point3DVct &o_points) {for (int i = 0; i < input_points.size(); ++i) {Point_3D p = input_points[i];o_points.push_back(p);}return true;
}bool KMeans::Cluster(const Point3DVct &input_points,std::vector<Point3DVct> &k_points_vct) {Point3DVct input_point3D_vct;SetInput(input_points, input_point3D_vct);Point3DVct v_center(k_center_point_vct_.size());do {for (size_t i = 0, pntCount = input_point3D_vct.size(); i < pntCount; ++i) {float min_dist = 10000000000;int point_class = 0;for (size_t j = 0; j < m_k; ++j) {float dist =DistBetweenPoints(input_point3D_vct[i], k_center_point_vct_[j]);if (min_dist - dist > 0.000001) {min_dist = dist;point_class = j;}}k_points_vct_[point_class].push_back(input_point3D_vct[i]);}//保存上一次迭代的中心点for (size_t i = 0; i < k_center_point_vct_.size(); ++i) {v_center[i] = k_center_point_vct_[i];}if (!UpdateGroupCenter(k_points_vct_, k_center_point_vct_)) {return false;}if (!ExistCenterShift(v_center, k_center_point_vct_)) {k_points_vct = k_points_vct_;break;}for (size_t i = 0; i < m_k; ++i) {for (int j = 0; j < k_points_vct_[i].size(); ++j) {const Point_3D &p = k_points_vct_[i][j];std::cout << "x= " << p.x << ",   y= " << p.y << ",   z= " << p.z<< " ,class: " << i << std::endl;}}std::cout << "--------------------- " << std::endl;for (size_t i = 0; i < m_k; ++i) {k_points_vct_[i].clear();}} while (true);return true;
}// 计算两个点之间的距离
float KMeans::DistBetweenPoints(const Point_3D &p1, const Point_3D &p2) {float dist = 0;float x_diff = 0, y_diff = 0, z_diff = 0;x_diff = p1.x - p2.x;y_diff = p1.y - p2.y;z_diff = p1.z - p2.z;dist = sqrt(x_diff * x_diff + y_diff * y_diff + z_diff * z_diff);return dist;
}bool KMeans::UpdateGroupCenter(std::vector<Point3DVct> &K_points_vct,Point3DVct &centers) {if (centers.size() != m_k) {std::cout << "类别的个数不为K" << std::endl;return false;}for (size_t i = 0; i < m_k; ++i) {float x = 0, y = 0, z = 0;size_t point_num_in_this_class = K_points_vct[i].size();// 遍历每个类别的数据,每次遍历都把一类数据的x全加起来,求平均数,赋值给该类别的中心x;// y全加起来,求平均数,赋值给该类别的中心y;// z全加起来,求平均数,赋值给该类别的中心zfor (size_t j = 0; j < point_num_in_this_class; ++j) {x += K_points_vct[i][j].x;y += K_points_vct[i][j].y;z += K_points_vct[i][j].z;}x /= point_num_in_this_class;y /= point_num_in_this_class;z /= point_num_in_this_class;centers[i].x = x;centers[i].y = y;centers[i].z = z;}return true;
}//是否存在中心点移动
// 就是说遍历K个类别的中心点,若上一次和本次更新的中心点距离变化大于一定值就表示正在更新更新了;
// 否则,就表示不再更新迭代停止;
// 只要有一个返回值大于阈值,就表示有数据更新,不能停止迭代。如果所有个类别的中心距离都小于某阈值,就表示更新停止.
bool KMeans::ExistCenterShift(Point3DVct &prev_center, Point3DVct &cur_center) {for (size_t i = 0; i < m_k; ++i) {float dist = DistBetweenPoints(prev_center[i], cur_center[i]);if (dist > DELTA) {return true;}}return false;
}

这篇关于【数学与算法】KMeans聚类代码的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1067567

相关文章

深入理解Mysql OnlineDDL的算法

《深入理解MysqlOnlineDDL的算法》本文主要介绍了讲解MysqlOnlineDDL的算法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小... 目录一、Online DDL 是什么?二、Online DDL 的三种主要算法2.1COPY(复制法)

Java集合之Iterator迭代器实现代码解析

《Java集合之Iterator迭代器实现代码解析》迭代器Iterator是Java集合框架中的一个核心接口,位于java.util包下,它定义了一种标准的元素访问机制,为各种集合类型提供了一种统一的... 目录一、什么是Iterator二、Iterator的核心方法三、基本使用示例四、Iterator的工

Java 线程池+分布式实现代码

《Java线程池+分布式实现代码》在Java开发中,池通过预先创建并管理一定数量的资源,避免频繁创建和销毁资源带来的性能开销,从而提高系统效率,:本文主要介绍Java线程池+分布式实现代码,需要... 目录1. 线程池1.1 自定义线程池实现1.1.1 线程池核心1.1.2 代码示例1.2 总结流程2. J

JS纯前端实现浏览器语音播报、朗读功能的完整代码

《JS纯前端实现浏览器语音播报、朗读功能的完整代码》在现代互联网的发展中,语音技术正逐渐成为改变用户体验的重要一环,下面:本文主要介绍JS纯前端实现浏览器语音播报、朗读功能的相关资料,文中通过代码... 目录一、朗读单条文本:① 语音自选参数,按钮控制语音:② 效果图:二、朗读多条文本:① 语音有默认值:②

Vue实现路由守卫的示例代码

《Vue实现路由守卫的示例代码》Vue路由守卫是控制页面导航的钩子函数,主要用于鉴权、数据预加载等场景,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着... 目录一、概念二、类型三、实战一、概念路由守卫(Navigation Guards)本质上就是 在路

uni-app小程序项目中实现前端图片压缩实现方式(附详细代码)

《uni-app小程序项目中实现前端图片压缩实现方式(附详细代码)》在uni-app开发中,文件上传和图片处理是很常见的需求,但也经常会遇到各种问题,下面:本文主要介绍uni-app小程序项目中实... 目录方式一:使用<canvas>实现图片压缩(推荐,兼容性好)示例代码(小程序平台):方式二:使用uni

JAVA实现Token自动续期机制的示例代码

《JAVA实现Token自动续期机制的示例代码》本文主要介绍了JAVA实现Token自动续期机制的示例代码,通过动态调整会话生命周期平衡安全性与用户体验,解决固定有效期Token带来的风险与不便,感兴... 目录1. 固定有效期Token的内在局限性2. 自动续期机制:兼顾安全与体验的解决方案3. 总结PS

C#中通过Response.Headers设置自定义参数的代码示例

《C#中通过Response.Headers设置自定义参数的代码示例》:本文主要介绍C#中通过Response.Headers设置自定义响应头的方法,涵盖基础添加、安全校验、生产实践及调试技巧,强... 目录一、基础设置方法1. 直接添加自定义头2. 批量设置模式二、高级配置技巧1. 安全校验机制2. 类型

Python屏幕抓取和录制的详细代码示例

《Python屏幕抓取和录制的详细代码示例》随着现代计算机性能的提高和网络速度的加快,越来越多的用户需要对他们的屏幕进行录制,:本文主要介绍Python屏幕抓取和录制的相关资料,需要的朋友可以参考... 目录一、常用 python 屏幕抓取库二、pyautogui 截屏示例三、mss 高性能截图四、Pill

使用MapStruct实现Java对象映射的示例代码

《使用MapStruct实现Java对象映射的示例代码》本文主要介绍了使用MapStruct实现Java对象映射的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,... 目录一、什么是 MapStruct?二、实战演练:三步集成 MapStruct第一步:添加 Mave