using System; using System.Collections.Generic; using System.Linq; using Collections.Pooled; using MoreLinq; namespace MoonTools.Core.Graph { public class DirectedWeightedMultiGraph : MultiGraph, IWeightedMultiGraph where TNode : IEquatable { protected Dictionary weights = new Dictionary(); public Guid AddEdge(TNode v, TNode u, int weight, TEdgeData data) { var id = BaseAddEdge(v, u, data); weights.Add(id, weight); return id; } public void AddEdges(params (TNode, TNode, int, TEdgeData)[] edges) { foreach (var edge in edges) { AddEdge(edge.Item1, edge.Item2, edge.Item3, edge.Item4); } } public override void Clear() { base.Clear(); weights.Clear(); } public IEnumerable Weights(TNode v, TNode u) { CheckNodes(v, u); return edges[(v, u)].Select(id => Weight(id)); } public int Weight(Guid id) { if (!IDToEdge.ContainsKey(id)) { throw new ArgumentException($"Edge with id {id} does not exist in the graph"); } return weights[id]; } private IEnumerable ReconstructPath(PooledDictionary cameFrom, TNode currentNode) { while (cameFrom.ContainsKey(currentNode)) { var edgeID = cameFrom[currentNode]; var edge = IDToEdge[edgeID]; currentNode = edge.Item1; yield return edgeID; } } public IEnumerable AStarShortestPath(TNode start, TNode end, Func heuristic) { CheckNodes(start, end); var openSet = new PooledSet(ClearMode.Always); var closedSet = new PooledSet(ClearMode.Always); var gScore = new PooledDictionary(ClearMode.Always); var fScore = new PooledDictionary(ClearMode.Always); var cameFrom = new PooledDictionary(ClearMode.Always); openSet.Add(start); gScore[start] = 0; fScore[start] = heuristic(start, end); while (openSet.Count > 0) { var currentNode = openSet.MinBy(node => fScore[node]).First(); if (currentNode.Equals(end)) { openSet.Dispose(); closedSet.Dispose(); gScore.Dispose(); fScore.Dispose(); foreach (var edgeID in ReconstructPath(cameFrom, currentNode).Reverse()) { yield return edgeID; } cameFrom.Dispose(); yield break; } openSet.Remove(currentNode); closedSet.Add(currentNode); foreach (var neighbor in Neighbors(currentNode)) { if (!closedSet.Contains(neighbor)) { var lowestEdgeID = EdgeIDs(currentNode, neighbor).MinBy(id => weights[id]).First(); var weight = weights[lowestEdgeID]; var tentativeGScore = gScore.ContainsKey(currentNode) ? gScore[currentNode] + weight : int.MaxValue; if (!openSet.Contains(neighbor) || tentativeGScore < gScore[neighbor]) { cameFrom[neighbor] = lowestEdgeID; gScore[neighbor] = tentativeGScore; fScore[neighbor] = tentativeGScore + heuristic(neighbor, end); openSet.Add(neighbor); } } } } openSet.Dispose(); closedSet.Dispose(); gScore.Dispose(); fScore.Dispose(); cameFrom.Dispose(); yield break; } private IEnumerable ShortestPath(TNode start, TNode end, Func> SSSPAlgorithm) { CheckNodes(start, end); var cameFrom = new PooledDictionary(ClearMode.Always); var reachable = new PooledSet(ClearMode.Always); foreach (var (node, previous, weight) in SSSPAlgorithm(start)) { cameFrom[node] = previous; reachable.Add(node); } if (!reachable.Contains(end)) { cameFrom.Dispose(); reachable.Dispose(); yield break; } foreach (var edge in ReconstructPath(cameFrom, end).Reverse()) { yield return edge; } cameFrom.Dispose(); reachable.Dispose(); } public IEnumerable<(TNode, Guid, int)> DijkstraSingleSourceShortestPath(TNode source) { if (weights.Values.Any(w => w < 0)) { throw new NegativeWeightException("Dijkstra cannot be used on a graph with negative edge weights. Try Bellman-Ford"); } CheckNodes(source); var distance = new PooledDictionary(ClearMode.Always); var previousEdgeIDs = new PooledDictionary(ClearMode.Always); foreach (var node in Nodes) { distance[node] = int.MaxValue; } distance[source] = 0; var q = Nodes.ToPooledList(); while (q.Count > 0) { var node = q.MinBy(n => distance[n]).First(); q.Remove(node); if (distance[node] == int.MaxValue) { break; } foreach (var neighbor in Neighbors(node)) { foreach (var edgeID in EdgeIDs(node, neighbor)) { var weight = Weight(edgeID); var alt = distance[node] + weight; if (alt < distance[neighbor]) { distance[neighbor] = alt; previousEdgeIDs[neighbor] = edgeID; } } } } foreach (var node in Nodes) { if (previousEdgeIDs.ContainsKey(node) && distance.ContainsKey(node)) { yield return (node, previousEdgeIDs[node], distance[node]); } } distance.Dispose(); previousEdgeIDs.Dispose(); } public IEnumerable DijkstraShortestPath(TNode start, TNode end) { return ShortestPath(start, end, DijkstraSingleSourceShortestPath); } public IEnumerable<(TNode, Guid, int)> BellmanFordSingleSourceShortestPath(TNode source) { CheckNodes(source); var distance = new PooledDictionary(ClearMode.Always); var previous = new PooledDictionary(ClearMode.Always); foreach (var node in Nodes) { distance[node] = int.MaxValue; } distance[source] = 0; for (int i = 0; i < Order; i++) { foreach (var edgeID in IDToEdge.Keys) { var weight = Weight(edgeID); var (v, u) = IDToEdge[edgeID]; if (distance[v] + weight < distance[u]) { distance[u] = distance[v] + weight; previous[u] = edgeID; } } } foreach (var edgeID in IDToEdge.Keys) { var (v, u) = IDToEdge[edgeID]; foreach (var weight in Weights(v, u)) { if (distance[v] + weight < distance[u]) { throw new NegativeCycleException(); } } } foreach (var node in Nodes) { if (previous.ContainsKey(node) && distance.ContainsKey(node)) { yield return (node, previous[node], distance[node]); } } distance.Dispose(); previous.Dispose(); } public IEnumerable BellmanFordShortestPath(TNode start, TNode end) { return ShortestPath(start, end, BellmanFordSingleSourceShortestPath); } } }