mirror of
https://github.com/trekhleb/javascript-algorithms.git
synced 2025-12-08 19:06:00 +00:00
108 lines
3.8 KiB
JavaScript
108 lines
3.8 KiB
JavaScript
import depthFirstSearch from '../depth-first-search/depthFirstSearch';
|
|
|
|
/**
|
|
* Helper class for visited vertex metadata.
|
|
*/
|
|
class VisitMetadata {
|
|
constructor({ discoveryTime, lowDiscoveryTime }) {
|
|
this.discoveryTime = discoveryTime;
|
|
this.lowDiscoveryTime = lowDiscoveryTime;
|
|
// We need this in order to check graph root node, whether it has two
|
|
// disconnected children or not.
|
|
this.independantChildrenCount = 0;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Tarjan's algorithm for rinding articulation points in graph.
|
|
*
|
|
* @param {Graph} graph
|
|
* @return {GraphVertex[]}
|
|
*/
|
|
export default function articulationPoints(graph) {
|
|
// Set of vertices we've already visited during DFS.
|
|
const visitedSet = {};
|
|
|
|
// Set of articulation points.
|
|
const articulationPointsSet = {};
|
|
|
|
// Time needed to discover to the current vertex.
|
|
let discoveryTime = 0;
|
|
|
|
// Peek the start vertex for DFS traversal.
|
|
const startVertex = graph.getAllVertices()[0];
|
|
|
|
const dfsCallbacks = {
|
|
/**
|
|
* @param {GraphVertex} currentVertex
|
|
* @param {GraphVertex} previousVertex
|
|
*/
|
|
enterVertex: ({ currentVertex, previousVertex }) => {
|
|
// Put current vertex to visited set.
|
|
visitedSet[currentVertex.getKey()] = new VisitMetadata({
|
|
discoveryTime,
|
|
lowDiscoveryTime: discoveryTime,
|
|
});
|
|
|
|
// Tick discovery time.
|
|
discoveryTime += 1;
|
|
|
|
if (previousVertex) {
|
|
// Update children counter for previous vertex.
|
|
visitedSet[previousVertex.getKey()].independantChildrenCount += 1;
|
|
}
|
|
},
|
|
/**
|
|
* @param {GraphVertex} currentVertex
|
|
* @param {GraphVertex} previousVertex
|
|
*/
|
|
leaveVertex: ({ currentVertex, previousVertex }) => {
|
|
if (previousVertex === null) {
|
|
// Don't do anything for the root vertex if it is already current (not previous one)
|
|
return;
|
|
}
|
|
|
|
// Update the low time with the smallest time of adjacent vertices.
|
|
// Get minimum low discovery time from all neighbors.
|
|
/** @param {GraphVertex} neighbor */
|
|
visitedSet[currentVertex.getKey()].lowDiscoveryTime = currentVertex.getNeighbors().reduce(
|
|
(lowestDiscoveryTime, neighbor) => {
|
|
const neighborLowTime = visitedSet[neighbor.getKey()].lowDiscoveryTime;
|
|
return neighborLowTime < lowestDiscoveryTime ? neighborLowTime : lowestDiscoveryTime;
|
|
},
|
|
visitedSet[currentVertex.getKey()].lowDiscoveryTime,
|
|
);
|
|
|
|
// Detect whether previous vertex is articulation point or not.
|
|
// To do so we need to check two [OR] conditions:
|
|
// 1. Is it a root vertex with at least two independent children.
|
|
// 2. If its visited time is <= low time of adjacent vertex.
|
|
if (previousVertex === startVertex) {
|
|
// Check that root vertex has at least two independent children.
|
|
if (visitedSet[previousVertex.getKey()].independantChildrenCount >= 2) {
|
|
articulationPointsSet[previousVertex.getKey()] = previousVertex;
|
|
}
|
|
} else {
|
|
// Get current vertex low discovery time.
|
|
const currentLowDiscoveryTime = visitedSet[currentVertex.getKey()].lowDiscoveryTime;
|
|
|
|
// Compare current vertex low discovery time with parent discovery time. Check if there
|
|
// are any short path (back edge) exists. If we can't get to current vertex other then
|
|
// via parent then the parent vertex is articulation point for current one.
|
|
const parentDiscoveryTime = visitedSet[previousVertex.getKey()].discoveryTime;
|
|
if (parentDiscoveryTime <= currentLowDiscoveryTime) {
|
|
articulationPointsSet[previousVertex.getKey()] = previousVertex;
|
|
}
|
|
}
|
|
},
|
|
allowTraversal: ({ nextVertex }) => {
|
|
return !visitedSet[nextVertex.getKey()];
|
|
},
|
|
};
|
|
|
|
// Do Depth First Search traversal over submitted graph.
|
|
depthFirstSearch(graph, startVertex, dfsCallbacks);
|
|
|
|
return Object.values(articulationPointsSet);
|
|
}
|