< Summary

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

Metrics

MethodBranch coverage Crap Score Cyclomatic complexity NPath complexity Sequence coverage
KMeansClusterer(...)0%2100%
KMeansClusterer(...)0%2100%
Cluster()0%72800%
AssignObjectsToCluster()0%20400%

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.
 011  private int _k = 0;
 12
 13  // Maximum number of iterations.
 014  private int _maxIterations = 20;
 15
 016  public KMeansClusterer(List<GameObject> objects, int k, int maxIterations = 20) : base(objects) {
 017    _k = k;
 018    _maxIterations = maxIterations;
 019  }
 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.
 026  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.
 029    System.Random random = new System.Random();
 030    for (int i = _objects.Count - 1; i >= _objects.Count - _k; --i) {
 031      int j = random.Next(i + 1);
 032      (_objects[i], _objects[j]) = (_objects[j], _objects[i]);
 033    }
 034    for (int i = _objects.Count - 1; i >= _objects.Count - _k; --i) {
 035      _clusters.Add(new Cluster(_objects[i]));
 036    }
 37
 038    bool converged = false;
 039    int iteration = 0;
 040    while (!converged && iteration < _maxIterations) {
 041      AssignObjectsToCluster();
 42
 43      // Calculate the new clusters as the mean of all assigned game objects.
 044      converged = true;
 045      for (int clusterIndex = 0; clusterIndex < _clusters.Count; ++clusterIndex) {
 46        Cluster newCluster;
 047        if (_clusters[clusterIndex].IsEmpty()) {
 048          int objectIndex = random.Next(_objects.Count);
 049          newCluster = new Cluster(_objects[objectIndex]);
 050        } else {
 051          newCluster = new Cluster(_clusters[clusterIndex].Centroid());
 052        }
 53
 54        // Check whether the algorithm has converged by checking whether the cluster has moved.
 055        if (Vector3.Distance(newCluster.Coordinates, _clusters[clusterIndex].Coordinates) >
 056            Epsilon) {
 057          converged = false;
 058        }
 59
 060        _clusters[clusterIndex] = newCluster;
 061      }
 62
 063      ++iteration;
 064    }
 65
 066    AssignObjectsToCluster();
 067  }
 68
 069  private void AssignObjectsToCluster() {
 70    // Determine the closest centroid to each game object.
 071    foreach (var obj in _objects) {
 072      float minDistance = Mathf.Infinity;
 073      int minIndex = -1;
 074      for (int clusterIndex = 0; clusterIndex < _clusters.Count; ++clusterIndex) {
 075        float distance =
 76            Vector3.Distance(_clusters[clusterIndex].Coordinates, obj.transform.position);
 077        if (distance < minDistance) {
 078          minDistance = distance;
 079          minIndex = clusterIndex;
 080        }
 081      }
 082      _clusters[minIndex].AddObject(obj);
 083    }
 084  }
 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}