机器学习之K means算法

Initialize k means with random values

For a given number of iterations:
    Iterate through items:
        Find the mean closest to the item
        Assign item to mean
        Update mean

The most common algorithm uses an iterative refinement technique. Due to its ubiquity it is often called the k-means algorithm; it is also referred to as Lloyd's algorithm, particularly in the computer science community.

Given an initial set of k means m1(1),…,mk(1) (see below), the algorithm proceeds by alternating between two steps:[6]

Assignment step: Assign each observation to the cluster whose mean has the least squared Euclidean distance, this is intuitively the "nearest" mean.[7] (Mathematically, this means partitioning the observations according to the Voronoi diagram generated by the means).
Si(t)={xp:‖xp−mi(t)‖2≤‖xp−mj(t)‖2 ∀j,1≤j≤k},S_{i}^{(t)}={\big \{}x_{p}:{\big \|}x_{p}-m_{i}^{(t)}{\big \|}^{2}\leq {\big \|}x_{p}-m_{j}^{(t)}{\big \|}^{2}\ \forall j,1\leq j\leq k{\big \}},
where each xpx_{p} is assigned to exactly one S(t)S^{(t)}, even if it could be assigned to two or more of them.
Update step: Calculate the new means to be the centroids of the observations in the new clusters.
mi(t+1)=1|Si(t)|∑xj∈Si(t)xjm_{i}^{(t+1)}={\frac {1}{|S_{i}^{(t)}|}}\sum _{x_{j}\in S_{i}^{(t)}}x_{j}

The algorithm has converged when the assignments no longer change. There is no guarantee that the optimum is found using this algorithm.[8]

// Author: Felix Duvallet

#include "kmeans.h"

#include <algorithm>
#include <cassert>
#include <fstream>
#include <iterator>
#include <random>
#include <sstream>
#include <string>
#include <utility>
#include <vector>

using namespace std;

KMeans::KMeans(int k, int max_iterations)
  : num_clusters_(k),
    max_iterations_(max_iterations) {
}

bool KMeans::init(const std::vector<Point> &points) {
  // Store all points and create a vector that looks like [0, 1, 2, ... N]
  std::vector<int> points_indices;
  int point_num = 0;
  points_.clear();

  for (const auto &p : points) {
    points_.push_back(p);
    points_indices.push_back(point_num);
    point_num += 1;
  }

  // Initialize the means randomly: shuffle the array of unique index
  // integers. This prevents assigning the mean to the same point twice.
  std::random_device rd;
  std::mt19937 rng(rd());
  std::shuffle(points_indices.begin(), points_indices.end(), rng);

  // Sanity check
  assert(points.size() >= num_clusters_);

  for (int idx = 0; idx < num_clusters_; ++idx) {
    Point mean(points[points_indices[idx]]);
    means_.push_back(mean);
  }

  return true;
}

bool KMeans::run() {
  for (int iteration = 1; iteration <= max_iterations_; ++iteration) {
    cout << "== KMeans iteration " << iteration << " == " << endl;
    bool changed = assign();
    update_means();

    if (!changed) {
      cout << "KMeans has converged after " << iteration <<  " iterations."
           << endl;
      return true;
    }
  }
  return false;
}

int KMeans::findNearestCluster(const Point &point) {
  double min_dist = 1e12;
  int min_cluster = -1;

  for (int idx = 0; idx < num_clusters_; ++idx) {
    const double dist = Point::distance(point, means_[idx]);
    if (dist < min_dist) {
      min_dist = dist;
      min_cluster = idx;
    }
  }

  return min_cluster;
}

bool KMeans::assign() {
  bool changed = false;
  // Assign each point to the nearest cluster: iterate over all points, find
  // the nearest cluster, and assign.
  for (auto &point : points_) {
    const int new_cluster = findNearestCluster(point);

    // set changed to true if the cluster was updated. Note that we cannot
    // inline (changed = changed || update) since the compiler will
    // 'optimize out' the update step.
    bool ret = point.update(new_cluster);
    changed = changed || ret;

    cout << "Assigned point " << point << " to cluster: "
    << new_cluster << endl;
  }
  return changed;
}

bool KMeans::update_means() {
  // Compute each mean as the mean of the points in that cluster.

  // First, compute a map of the cluster assignments. This prevents us from
  // iterating over the data k times.
  std::multimap<int, const Point *> point_cluster_map;
  for (const auto &point : points_) {
    // Map is cluster_index -> Point*
    auto pair = std::pair<int, const Point *>(point.cluster_, &point);
    point_cluster_map.insert(pair);
  }

  // Iterate over each cluster, computing the mean.
  for (int cluster_idx = 0; cluster_idx < num_clusters_; ++cluster_idx) {
    computeClusterMean(point_cluster_map, cluster_idx,
                       &means_[cluster_idx]);
  }
  return true;
}

void KMeans::computeClusterMean(
  const std::multimap<int, const Point *> &multimap,
  int cluster,
  Point *mean) {
  // Zero-out the mean.
  for (int dim = 0; dim < mean->dimensions_; ++dim)
    mean->data_[dim] = 0.0;

  // Find all the points in the given cluster, this returns an iterator pair
  // (begin and end).
  auto in_cluster = multimap.equal_range(cluster);
  int num_points = 0;

  // Compute the mean: sum over all points, then divide by the number.
  for (auto itr = in_cluster.first; itr != in_cluster.second; ++itr) {
    mean->add(*(itr->second));

    num_points += 1;
  }

  for (unsigned int idx = 0; idx < mean->dimensions_; ++idx) {
    mean->data_[idx] /= float(num_points);
  }
}

void KMeans::printMeans() {
  for (const auto &mean : means_) {
    cout << "Mean: " << mean << endl;
  }
}

// static
bool KMeans::loadPoints(const string &filepath, vector<Point> *points) {
  std::ifstream file_stream(filepath, std::ios_base::in);
  if (!file_stream) {
    cout << "Could not open file " << filepath << endl;
    return false;
  }

  std::string line;
  // Split up each line of the file.
  while (getline(file_stream, line, '\n')) {
    std::stringstream line_stream(line);

    // Get a vector of numbers directly from a stream iterator.
    std::istream_iterator<double> start(line_stream), end;
    std::vector<double> numbers(start, end);

    Point p(numbers);
    points->push_back(p);
  }

  return true;
}

void KMeans::writeMeans(const std::string &filepath) {
  std::ofstream file_stream(filepath, std::ios_base::out);
  if (!file_stream) {
    cout << "Could not open file " << filepath << endl;
    return;
  }

  // Copy all data to file_stream, then append a newline.
  for (auto &mean : means_) {
    std::ostream_iterator<double> itr(file_stream, " ");
    std::copy(mean.data_.begin(), mean.data_.end(), itr);
    file_stream << endl;
  }
  return;

}

 

posted @ 2018-02-09 01:34  jxr041100  阅读(308)  评论(0编辑  收藏  举报