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聚类代码相关推荐

  1. 机器学习算法-KMeans聚类算法解析及伪代码实现。

    机器学习算法-KMeans聚类算法解析及伪代码实现. 徐小狗在文末附上了几条大神们关于KMeans聚类算法的博文,欲详细研究请前往浏览~ 作为初学者,许多地方可能笨拙或有误,希望有大神看到后给予优化和 ...

  2. MATLAB K-means聚类代码讲解

    一.概述 K-means聚类采用类内距离和最小的方式对数据分类,MATLAB中自带K-means算法,最简单的调用如下: idx=kmeans(x,k) 将n-by-p数据矩阵x中的数据划分为k个类簇 ...

  3. 03 聚类算法 - K-means聚类

    02 聚类算法 - 相似度距离公式.维度灾难 二.K-means聚类 给定一个有M个对象的数据集,构建一个具有k个__簇__的模型,其中__k<=M__.满足以下条件: 1.每个簇至少包含一个对 ...

  4. 大数据进阶之算法——KMeans聚类算法

    首先说一下分类和聚类的区别: 分类: 分类其实就是从特定的数据中挖掘模式,做出相对应的判断.例如对班级的学生进性性别的分类,我事先已经知道只有男性和女性两个分类. 聚类: 聚类的目的也是将数据分类,但 ...

  5. (学习笔记)十大经典算法——K-means聚类算法

    概述 聚类算法是在无监督的情况下将对象自动分组的一种分析方法,典型的聚类算法分为三个阶段:特征选择和特征提取,数据对象间相似度计算,根据相似度将数据对象分组.聚类算法的目标是将数据集合分成若干簇,使得 ...

  6. kmeans聚类代码详解

    kmeans聚类 keans算法主要是将所有点与质心算距离,与每个点距离最小的质心就是它的类别,然后再从每一类中选取平均值作为新的质心 #核心算法中的数据处理示例: #假设五个点与四个之心距离分别如下 ...

  7. 数据分析——算法——K-means聚类(天池:汽车产品聚类分析)

    K-means聚类 目录 K-means聚类 1 简介 2 Python实战 1 简介 原理:通过计算不同样本间的距离来判断他们的相近关系的,相近的就会放到同一个类别中去. 适用数据:数值数据 优点: ...

  8. matlab中的聚类算法,kmeans聚类算法matlab matlab 聚类算法silhouette

    怎样用matlab实现多维K-means聚类算法小编觉得一个好的周末应该是这样的:睡到中午醒来,在床上躺着玩两个小时手机,起床随便吃点东西,下午去超市买一大堆零食,五六点的时候去约小伙伴们吃火锅烧烤, ...

  9. 机器学习算法-k-means聚类算法介绍

    聚类算法介绍: 在"无监督学习"(unsupervised learning)中,训练样本的标记信息是未知的,目标是通过对无标记训练样本的学习来揭示数据的内在性质及规律,为进一步的 ...

最新文章

  1. Only Train Once:微软、浙大等研究者提出剪枝框架OTO,无需微调即可获得轻量级架构...
  2. linux注册硬盘中断,基于Linux底层硬盘中断辅助代码的分析与设计
  3. webdav 软件_6 款功能强大的工具软件,有一款人人必备
  4. 【每日SQL打卡】​​​​​​​​​​​​​​​DAY 24丨找到连续区间的开始和结束数字【难度中等】​
  5. 9、1.4.1 JDK下载与安装
  6. python 如何加密_Python如何玩转加密?
  7. sqlyog的快捷键
  8. Linux-四-常见符号
  9. 基于springboot的美食点评APP设计与实现
  10. wifi连接一段时间才能上网_家里的WiFi老是突然断网,WiFi还是连着的,但是关WiFi又连不上去,必须重启路由器才能上网...
  11. 浅析SEO搜索引擎优化
  12. 自定义View之滚动刻度尺,2018/1/14 05
  13. 都说学3D建模赚钱,到底学了能做什么?
  14. 计算机视觉 响应_视觉响应式布局的自动化测试
  15. Cadence Allegro PCB添加泪滴的方法
  16. pyinstaller说明(windows、mac、linux)
  17. Mkz-Cloud 部署之路
  18. 虚拟机克隆以及网络配置教程
  19. python会员折扣_Python——潜在会员用户预测
  20. 交换机日志删除_锐捷交换机如何删除日志

热门文章

  1. 第九章 组合模型在信贷风控中的应用
  2. spring boot自动配置之jdbc
  3. SolrCloud之分布式索引及与Zookeeper的集成--转载
  4. Redis双机热备方案--转
  5. android 组件(activity,service,content provider,broadcast receiver,intent)详解
  6. 人脸识别技术大总结—Face Detection Alignment
  7. 独立开发者:新手做2D手游该用哪些工具?
  8. 高德地图横屏不显示服务器,高德地图不能横屏!
  9. SpringBoot - 优雅的实现【业务校验】高级进阶
  10. MySQL-数据库监控初探