< Summary

Class:KMeansClusterer
Assembly:bamlab.micromissiles
File(s):/github/workspace/Assets/Scripts/Algorithms/Clustering/KMeansClusterer.cs
Covered lines:49
Uncovered lines:6
Coverable lines:55
Total lines:123
Line coverage:89% (49 of 55)
Covered branches:0
Total branches:0
Covered methods:4
Total methods:4
Method coverage:100% (4 of 4)

Metrics

MethodBranch coverage Crap Score Cyclomatic complexity NPath complexity Sequence coverage
KMeansClusterer(...)0%110100%
KMeansClusterer(...)0%2100%
Cluster()0%8.058090.7%
AssignObjectsToCluster()0%440100%

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.
 1011  private int _k = 0;
 12
 13  // Maximum number of iterations.
 1014  private int _maxIterations = 20;
 15
 2016  public KMeansClusterer(List<GameObject> objects, int k, int maxIterations = 20) : base(objects) {
 1017    _k = k;
 1018    _maxIterations = maxIterations;
 1019  }
 020  public KMeansClusterer(List<Agent> agents, int k, int maxIterations = 20) : base(agents) {
 021    _k = k;
 022    _maxIterations = maxIterations;
 023  }
 24
 25  // Cluster the game objects.
 1026  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.
 1029    System.Random random = new System.Random();
 8330    for (int i = _objects.Count - 1; i >= _objects.Count - _k; --i) {
 2131      int j = random.Next(i + 1);
 2132      (_objects[i], _objects[j]) = (_objects[j], _objects[i]);
 2133    }
 8334    for (int i = _objects.Count - 1; i >= _objects.Count - _k; --i) {
 2135      _clusters.Add(new Cluster(_objects[i]));
 2136    }
 37
 1038    bool converged = false;
 1039    int iteration = 0;
 4840    while (!converged && iteration < _maxIterations) {
 1941      AssignObjectsToCluster();
 42
 43      // Calculate the new clusters as the mean of all assigned game objects.
 1944      converged = true;
 14645      for (int clusterIndex = 0; clusterIndex < _clusters.Count; ++clusterIndex) {
 46        Cluster newCluster;
 3647        if (_clusters[clusterIndex].IsEmpty()) {
 048          int objectIndex = random.Next(_objects.Count);
 049          newCluster = new Cluster(_objects[objectIndex]);
 3650        } else {
 3651          newCluster = new Cluster(_clusters[clusterIndex].Centroid());
 3652        }
 53
 54        // Check whether the algorithm has converged by checking whether the cluster has moved.
 3655        if (Vector3.Distance(newCluster.Coordinates, _clusters[clusterIndex].Coordinates) >
 1156            Epsilon) {
 1157          converged = false;
 1158        }
 59
 3660        _clusters[clusterIndex] = newCluster;
 3661      }
 62
 1963      ++iteration;
 1964    }
 65
 1066    AssignObjectsToCluster();
 1067  }
 68
 2969  private void AssignObjectsToCluster() {
 70    // Determine the closest centroid to each game object.
 48371    foreach (var obj in _objects) {
 13272      float minDistance = Mathf.Infinity;
 13273      int minIndex = -1;
 104474      for (int clusterIndex = 0; clusterIndex < _clusters.Count; ++clusterIndex) {
 26075        float distance =
 76            Vector3.Distance(_clusters[clusterIndex].Coordinates, obj.transform.position);
 45077        if (distance < minDistance) {
 19078          minDistance = distance;
 19079          minIndex = clusterIndex;
 19080        }
 26081      }
 13282      _clusters[minIndex].AddObject(obj);
 13283    }
 2984  }
 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)
 91      : base(objects, maxSize, maxRadius) {}
 92
 93  // Cluster the game objects.
 94  public override void Cluster() {
 95    int numClusters = (int)Mathf.Ceil(_objects.Count / _maxSize);
 96    KMeansClusterer clusterer;
 97    while (true) {
 98      clusterer = new KMeansClusterer(_objects, numClusters);
 99      clusterer.Cluster();
 100
 101      // Count the number of over-populated and over-sized clusters.
 102      int numOverPopulatedClusters = 0;
 103      int numOverSizedClusters = 0;
 104      foreach (var cluster in clusterer.Clusters) {
 105        if (cluster.Size() > _maxSize) {
 106          ++numOverPopulatedClusters;
 107        }
 108        if (cluster.Radius() > _maxRadius) {
 109          ++numOverSizedClusters;
 110        }
 111      }
 112
 113      // If all clusters satisfy the size and radius constraints, the algorithm has converged.
 114      if (numOverPopulatedClusters == 0 && numOverSizedClusters == 0) {
 115        break;
 116      }
 117
 118      numClusters +=
 119          (int)Mathf.Ceil(Mathf.Max(numOverPopulatedClusters, numOverSizedClusters) / 2.0f);
 120    }
 121    _clusters = new List<Cluster>(clusterer.Clusters);
 122  }
 123}