< Summary

Class:AgglomerativeClusterer
Assembly:bamlab.micromissiles
File(s):/github/workspace/Assets/Scripts/Algorithms/Clustering/AgglomerativeClusterer.cs
Covered lines:63
Uncovered lines:1
Coverable lines:64
Total lines:101
Line coverage:98.4% (63 of 64)
Covered branches:0
Total branches:0
Covered methods:2
Total methods:3
Method coverage:66.6% (2 of 3)

Metrics

MethodBranch coverage Crap Score Cyclomatic complexity NPath complexity Sequence coverage
AgglomerativeClusterer(...)0%110100%
AgglomerativeClusterer(...)0%2100%
Cluster()0%18180100%

File(s)

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

#LineLine coverage
 1using System.Collections;
 2using System.Collections.Generic;
 3using UnityEngine;
 4
 5// The agglomerative clusterer class performs agglomerative clustering with the stopping condition
 6// given by the size and radius constraints.
 7public class AgglomerativeClusterer : ISizeAndRadiusConstrainedClusterer {
 8  public AgglomerativeClusterer(List<GameObject> objects, int maxSize, float maxRadius)
 129      : base(objects, maxSize, maxRadius) {}
 10  public AgglomerativeClusterer(List<Agent> agents, int maxSize, float maxRadius)
 011      : base(agents, maxSize, maxRadius) {}
 12
 13  // Cluster the game objects.
 414  public override void Cluster() {
 15    // Add a cluster for every game object.
 6016    foreach (var obj in _objects) {
 1617      Cluster cluster = new Cluster(obj);
 1618      cluster.AddObject(obj);
 1619      _clusters.Add(cluster);
 1620    }
 21
 22    // Create a set containing all valid cluster indices.
 423    HashSet<int> validClusterIndices = new HashSet<int>();
 5624    for (int i = 0; i < _clusters.Count; ++i) {
 1625      validClusterIndices.Add(i);
 1626    }
 27
 28    // Find the pairwise distances between all clusters.
 29    // The upper triangular half of the distances matrix is unused.
 430    float[,] distances = new float[_clusters.Count, _clusters.Count];
 5631    for (int i = 0; i < _clusters.Count; ++i) {
 10432      for (int j = 0; j < i; ++j) {
 2433        distances[i, j] = Vector3.Distance(_clusters[i].Coordinates, _clusters[j].Coordinates);
 2434      }
 1635    }
 36
 2837    while (true) {
 38      // Find the minimum distance between two clusters.
 1439      float minDistance = Mathf.Infinity;
 1440      int clusterIndex1 = -1;
 1441      int clusterIndex2 = -1;
 19642      for (int i = 0; i < _clusters.Count; ++i) {
 36443        for (int j = 0; j < i; ++j) {
 10244          if (distances[i, j] < minDistance) {
 1845            minDistance = distances[i, j];
 1846            clusterIndex1 = i;
 1847            clusterIndex2 = j;
 1848          }
 8449        }
 5650      }
 51
 52      // Check whether the minimum distance exceeds the maximum cluster radius, in which case the
 53      // algorithm has converged. This produces a conservative solution because the radius of a
 54      // merged cluster is less than or equal to the sum of the original cluster radii.
 1855      if (minDistance >= _maxRadius) {
 456        break;
 57      }
 58
 59      // Check whether merging the two clusters would violate the size constraint.
 1660      if (_clusters[clusterIndex1].Size() + _clusters[clusterIndex2].Size() > _maxSize) {
 661        distances[clusterIndex1, clusterIndex2] = Mathf.Infinity;
 662        continue;
 63      }
 64
 65      // Merge the two clusters together.
 466      int minClusterIndex = Mathf.Min(clusterIndex1, clusterIndex2);
 467      int maxClusterIndex = Mathf.Max(clusterIndex1, clusterIndex2);
 468      _clusters[minClusterIndex].Merge(_clusters[maxClusterIndex]);
 469      _clusters[minClusterIndex].Recenter();
 470      validClusterIndices.Remove(maxClusterIndex);
 71
 72      // Update the distances matrix using the distance between the cluster centroids.
 73      // TODO(titan): Change the distance metric to use average or maximum linkage.
 1474      for (int i = 0; i < minClusterIndex; ++i) {
 475        if (distances[minClusterIndex, i] < Mathf.Infinity) {
 276          distances[minClusterIndex, i] =
 77              Vector3.Distance(_clusters[minClusterIndex].Coordinates, _clusters[i].Coordinates);
 278        }
 279      }
 3880      for (int i = minClusterIndex + 1; i < _clusters.Count; ++i) {
 1781        if (distances[i, minClusterIndex] < Mathf.Infinity) {
 782          distances[i, minClusterIndex] =
 83              Vector3.Distance(_clusters[minClusterIndex].Coordinates, _clusters[i].Coordinates);
 784        }
 1085      }
 3286      for (int i = 0; i < maxClusterIndex; ++i) {
 887        distances[maxClusterIndex, i] = Mathf.Infinity;
 888      }
 2089      for (int i = maxClusterIndex + 1; i < _clusters.Count; ++i) {
 490        distances[i, maxClusterIndex] = Mathf.Infinity;
 491      }
 492    }
 93
 94    // Select only the valid clusters.
 5695    for (int i = _clusters.Count - 1; i >= 0; --i) {
 2096      if (!validClusterIndices.Contains(i)) {
 497        _clusters.RemoveAt(i);
 498      }
 1699    }
 4100  }
 101}