# Critical Connections in a Network | Tarjan's Algorithm

# Problem Statement:-

There are n servers numbered from 0 to n - 1 connected by undirected server-to-server connections forming a network where connections[i] = [ai, bi] represents a connection between servers ai and bi. Any server can reach other servers directly or indirectly through the network.

A critical connection is a connection that, if removed, will make some servers unable to reach some other server.

Return all critical connections in the network in any order.

Example 1:- Input: n = 4, connections = [[0,1],[1,2],[2,0],[1,3]]

Output: [[1,3]]

Explanation: [[3,1]] is also accepted.

# Approach

This problem requires us to find critical connections in a network, removing which some server might not be able to reach some other server. The gist is we have to find bridges in an undirected graph. What we can use for that? Tarjan's Algorithm!

What is Tarjan's Algorithm? Tarjan's Algorithm helps us to find strongly connected components in a graph, articulation points and bridges in a graph. Here we keep track of parent of every node, the lowlink value for every node and the id of the node. Initially, we intialise all the parents, lowlink values and id of every node as -1. When the id of a node is -1, it means that the node is not yet visited.

par[node] => Denotes the parent of the node id[node] => Denotes the id of the node low[node] => Denotes the lowlink value of the node, i.e., lowest id number of the nodes associated with it time => Denotes the time at which node gets visited (1st node gets visited at 0, it's neighbours get visited at 1 and so on..)

When we visit a node, we assign the lowlink value and id number to it. Since the id number is no more negative, it also shows that node is visited now. Now we traverse the adjacent nodes of the given node.

If any adjacent (u) is unvisited, we mark this node as it's (u)'s parent and perform a DFS on it. After that we minimise the given node's lowlink value with this adjacent node(u)'s lowlink value. If the id number of the given node is lesser than lowlink value of its adjacent node, we have found a bridge!

If the adjacent is visited, we check if its not the parent of given node as parent is also adjacent of node, and if it's not, we minimise the lowlink value of node with id number of the adjacent. And we are done!

```
class Solution {
public:
vector<vector<int>> ans; // required answer
vector<vector<int>> adj; // undirected graph
vector<int> id; // id number
vector<int> low; // lowlink value
vector<int> par; // parent
int time=0; // time
void dfs(int node){
id[node]=low[node]=time++; // incrementing time after each node gets visited
for(int &u:adj[node]) //exploring all the adjacent nodes of node
{
if(id[u]==-1) // checking if adjacent node is unvisited
{
par[u]=node; // assigning node as parent of adjacent node
dfs(u); // performing dfs for adjacent node
low[node]=min(low[node],low[u]); // minimising lowlink value of node
if(id[node]<low[u]) // checking if we have a bridge yet
ans.push_back({node,u});
}
else if(u!=par[node]) // checking if adjacent is not parent of node
low[node]=min(low[node],id[u]); // minimising lowlink value of node
}
}
vector<vector<int>> criticalConnections(int n, vector<vector<int>>& connections) {
adj.resize(n);
par.resize(n,-1);
low.resize(n,-1);
id.resize(n,-1);
for(int i=0;i<connections.size();i++){ // building our graph
adj[connections[i][0]].push_back(connections[i][1]);
adj[connections[i][1]].push_back(connections[i][0]);
}
for(int i=0;i<n;i++) // performing dfs on unvisited nodes
{
if(id[i]==-1)
dfs(i);
}
return ans;
}
};
```

Time Complexity: O(V+E) (No. of Vertices + No. of Edges)