dist-javascript-algorithms-and-data-structures
Version:
Algorithms and data-structures implemented on JavaScript
107 lines (87 loc) • 3.57 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = graphBridges;
var _depthFirstSearch = _interopRequireDefault(require("../depth-first-search/depthFirstSearch"));
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
/**
* Helper class for visited vertex metadata.
*/
class VisitMetadata {
constructor({
discoveryTime,
lowDiscoveryTime
}) {
this.discoveryTime = discoveryTime;
this.lowDiscoveryTime = lowDiscoveryTime;
}
}
/**
* @param {Graph} graph
* @return {Object}
*/
function graphBridges(graph) {
// Set of vertices we've already visited during DFS.
const visitedSet = {}; // Set of bridges.
const bridges = {}; // 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
*/
enterVertex: ({
currentVertex
}) => {
// Tick discovery time.
discoveryTime += 1; // Put current vertex to visited set.
visitedSet[currentVertex.getKey()] = new VisitMetadata({
discoveryTime,
lowDiscoveryTime: discoveryTime
});
},
/**
* @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;
} // Check if current node is connected to any early node other then previous one.
visitedSet[currentVertex.getKey()].lowDiscoveryTime = currentVertex.getNeighbors().filter(earlyNeighbor => earlyNeighbor.getKey() !== previousVertex.getKey()).reduce(
/**
* @param {number} lowestDiscoveryTime
* @param {GraphVertex} neighbor
*/
(lowestDiscoveryTime, neighbor) => {
const neighborLowTime = visitedSet[neighbor.getKey()].lowDiscoveryTime;
return neighborLowTime < lowestDiscoveryTime ? neighborLowTime : lowestDiscoveryTime;
}, visitedSet[currentVertex.getKey()].lowDiscoveryTime); // Compare low discovery times. In case if current low discovery time is less than the one
// in previous vertex then update previous vertex low time.
const currentLowDiscoveryTime = visitedSet[currentVertex.getKey()].lowDiscoveryTime;
const previousLowDiscoveryTime = visitedSet[previousVertex.getKey()].lowDiscoveryTime;
if (currentLowDiscoveryTime < previousLowDiscoveryTime) {
visitedSet[previousVertex.getKey()].lowDiscoveryTime = currentLowDiscoveryTime;
} // 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) {
const bridge = graph.findEdge(previousVertex, currentVertex);
bridges[bridge.getKey()] = bridge;
}
},
allowTraversal: ({
nextVertex
}) => {
return !visitedSet[nextVertex.getKey()];
}
}; // Do Depth First Search traversal over submitted graph.
(0, _depthFirstSearch.default)(graph, startVertex, dfsCallbacks);
return bridges;
}