684: Redundant-Connection
Medium

My algorithm is quite dependent on depth-first search (again).

My understanding of the question was that you are trying to remove an edge from a cycle. Unfortunately, it cannot be any edge from the cycle; it has to be the one that appears last in edges. This means, we have to somehow keep track of all the nodes present in the cycle.

In reality, this is just a slightly augmented DFS algorithm. The “simple” “top-level” overview is that you just DFS until you find a duplicate node that is not the parent node. Then, you consequently backtrack until you reach the first copy of the duplicate node. Afterwards, you continue to backtrack until you reach the first node, but you have to remove every node you visit. This will result in you having only all the elements of the cycle stored.

Finally, just iterate through edges and track the last index, where edges[0] and edges[1] are both present in the cycle.

Code:

class Solution {
private:
    int dfs(int parent, int child, vector<int> &seen, vector<vector<int>> &adjacencyMatrix) {
        seen[child] = 1;
        for (int i = 0; i < adjacencyMatrix[child].size(); ++i) {
            if (parent == adjacencyMatrix[child][i]) continue;
            if (seen[adjacencyMatrix[child][i]] == 1 && parent != adjacencyMatrix[child][i]) {
                return adjacencyMatrix[child][i];
            }
            int temp = dfs(child, adjacencyMatrix[child][i], seen, adjacencyMatrix);
            if (temp == -2) {
                seen[child] = 0;
                return -2;
            } else if (temp != -1) {
                if (temp == child) {
                    return -2;
                }
                else {
                    return temp;
                }
            } 
        }
        seen[child] = 0;
        return -1;
    }
public:
    vector<int> findRedundantConnection(vector<vector<int>>& edges) {
        vector<vector<int>> adjacencyMatrix (edges.size()); 

        for (int i = 0; i < edges.size(); ++i) {
            adjacencyMatrix[edges[i][0]-1].push_back(edges[i][1]-1);
            adjacencyMatrix[edges[i][1]-1].push_back(edges[i][0]-1);
        }

        vector<int> seen (edges.size(), 0);
        seen[0] = 1;
        for (int i = 0; i < adjacencyMatrix[0].size(); ++i) {
            int temp = dfs(0, int(adjacencyMatrix[0][i]), seen, adjacencyMatrix);
            if (temp == -2) {
                seen[0] = 0;
            }
        }

        int index = 0;
        for (int i = 0; i < edges.size(); ++i) {
            if (seen[edges[i][0]-1] == 1 and seen[edges[i][1]-1] == 1) {
                index = i;
            }
        }

        return edges[index];
   }
};

Complexity: