< Summary

Class:ConstrainedKMeansClusterer
Assembly:bamlab.micromissiles
File(s):/github/workspace/Assets/Scripts/Algorithms/Clustering/KMeansClusterer.cs
Covered lines:0
Uncovered lines:22
Coverable lines:22
Total lines:123
Line coverage:0% (0 of 22)
Covered branches:0
Total branches:0
Covered methods:0
Total methods:2
Method coverage:0% (0 of 2)

Metrics

MethodBranch coverage Crap Score Cyclomatic complexity NPath complexity Sequence coverage
ConstrainedKMeansClusterer(...)0%2100%
Cluster()0%56700%

File(s)

/github/workspace/Assets/Scripts/Algorithms/Clustering/KMeansClusterer.cs

#LineLine coverage
 1using System;
 2using System.Collections;
 3using System.Collections.Generic;
 4using UnityEngine;
 5
 6// The k-means clusterer class performs k-means clustering.
 7public class KMeansClusterer : IClusterer {
 8  public const float Epsilon = 1e-3f;
 9
 10  // Number of clusters.
 11  private int _k = 0;
 12
 13  // Maximum number of iterations.
 14  private int _maxIterations = 20;
 15
 16  public KMeansClusterer(List<GameObject> objects, int k, int maxIterations = 20) : base(objects) {
 17    _k = k;
 18    _maxIterations = maxIterations;
 19  }
 20  public KMeansClusterer(List<Agent> agents, int k, int maxIterations = 20) : base(agents) {
 21    _k = k;
 22    _maxIterations = maxIterations;
 23  }
 24
 25  // Cluster the game objects.
 26  public override void Cluster() {
 27    // Initialize the clusters with centroids located at random game objects.
 28    // Perform Fisher-Yates shuffling to find k random game objects.
 29    System.Random random = new System.Random();
 30    for (int i = _objects.Count - 1; i >= _objects.Count - _k; --i) {
 31      int j = random.Next(i + 1);
 32      (_objects[i], _objects[j]) = (_objects[j], _objects[i]);
 33    }
 34    for (int i = _objects.Count - 1; i >= _objects.Count - _k; --i) {
 35      _clusters.Add(new Cluster(_objects[i]));
 36    }
 37
 38    bool converged = false;
 39    int iteration = 0;
 40    while (!converged && iteration < _maxIterations) {
 41      AssignObjectsToCluster();
 42
 43      // Calculate the new clusters as the mean of all assigned game objects.
 44      converged = true;
 45      for (int clusterIndex = 0; clusterIndex < _clusters.Count; ++clusterIndex) {
 46        Cluster newCluster;
 47        if (_clusters[clusterIndex].IsEmpty()) {
 48          int objectIndex = random.Next(_objects.Count);
 49          newCluster = new Cluster(_objects[objectIndex]);
 50        } else {
 51          newCluster = new Cluster(_clusters[clusterIndex].Centroid());
 52        }
 53
 54        // Check whether the algorithm has converged by checking whether the cluster has moved.
 55        if (Vector3.Distance(newCluster.Coordinates, _clusters[clusterIndex].Coordinates) >
 56            Epsilon) {
 57          converged = false;
 58        }
 59
 60        _clusters[clusterIndex] = newCluster;
 61      }
 62
 63      ++iteration;
 64    }
 65
 66    AssignObjectsToCluster();
 67  }
 68
 69  private void AssignObjectsToCluster() {
 70    // Determine the closest centroid to each game object.
 71    foreach (var obj in _objects) {
 72      float minDistance = Mathf.Infinity;
 73      int minIndex = -1;
 74      for (int clusterIndex = 0; clusterIndex < _clusters.Count; ++clusterIndex) {
 75        float distance =
 76            Vector3.Distance(_clusters[clusterIndex].Coordinates, obj.transform.position);
 77        if (distance < minDistance) {
 78          minDistance = distance;
 79          minIndex = clusterIndex;
 80        }
 81      }
 82      _clusters[minIndex].AddObject(obj);
 83    }
 84  }
 85}
 86
 87// The constrained k-means clusterer class performs k-means clustering under size and radius
 88// constraints.
 89public class ConstrainedKMeansClusterer : ISizeAndRadiusConstrainedClusterer {
 90  public ConstrainedKMeansClusterer(List<GameObject> objects, int maxSize, float maxRadius)
 091      : base(objects, maxSize, maxRadius) {}
 92
 93  // Cluster the game objects.
 094  public override void Cluster() {
 095    int numClusters = (int)Mathf.Ceil(_objects.Count / _maxSize);
 96    KMeansClusterer clusterer;
 097    while (true) {
 098      clusterer = new KMeansClusterer(_objects, numClusters);
 099      clusterer.Cluster();
 100
 101      // Count the number of over-populated and over-sized clusters.
 0102      int numOverPopulatedClusters = 0;
 0103      int numOverSizedClusters = 0;
 0104      foreach (var cluster in clusterer.Clusters) {
 0105        if (cluster.Size() > _maxSize) {
 0106          ++numOverPopulatedClusters;
 0107        }
 0108        if (cluster.Radius() > _maxRadius) {
 0109          ++numOverSizedClusters;
 0110        }
 0111      }
 112
 113      // If all clusters satisfy the size and radius constraints, the algorithm has converged.
 0114      if (numOverPopulatedClusters == 0 && numOverSizedClusters == 0) {
 0115        break;
 116      }
 117
 0118      numClusters +=
 119          (int)Mathf.Ceil(Mathf.Max(numOverPopulatedClusters, numOverSizedClusters) / 2.0f);
 0120    }
 0121    _clusters = new List<Cluster>(clusterer.Clusters);
 0122  }
 123}