The Depth-First Search (DFS) approach is one of the most intuitive ways to solve the graph cloning problem. Here, we will recursively explore each node starting from the root (Node 1) and keep a map to store already cloned nodes, ensuring each node is cloned once.
For each node, we:
Time Complexity: O(V+E), where V is the number of vertices and E is the number of edges. This is because we visit each node and edge once.
Space Complexity: O(V), for the recursion stack and the clone map.
1#include <stdio.h>
2#include <stdlib.h>
3#include <string.h>
4
5#define MAX_NODES 101
6
7typedef struct Node {
8 int val;
9 struct Node** neighbors;
10 int numNeighbors;
11} Node;
12
13Node* cloneGraph(Node* node, Node** cloneMap);
14
15Node* cloneGraphUtil(Node* node, Node** cloneMap) {
16 if (node == NULL) return NULL;
17
18 if (cloneMap[node->val]) return cloneMap[node->val];
19
20 Node* cloneNode = (Node*)malloc(sizeof(Node));
21 cloneNode->val = node->val;
22 cloneNode->numNeighbors = node->numNeighbors;
23 cloneNode->neighbors = (Node**)malloc(sizeof(Node*) * node->numNeighbors);
24 cloneMap[node->val] = cloneNode;
25
26 for (int i = 0; i < node->numNeighbors; i++) {
27 cloneNode->neighbors[i] = cloneGraphUtil(node->neighbors[i], cloneMap);
28 }
29
30 return cloneNode;
31}
32
33Node* cloneGraph(Node* node) {
34 Node* cloneMap[MAX_NODES] = {NULL};
35 return cloneGraphUtil(node, cloneMap);
36}
The above C program defines a structure Node
for graph nodes and uses a recursive function cloneGraphUtil
to perform a DFS for cloning. We use an array of pointers cloneMap
indexed by node values to track cloned nodes.
An alternative approach is to use Breadth-First Search (BFS), which is iterative in nature. Here, we utilize a queue to help explore each node level by level, preventing deep recursion and managing each node's clone in a breadth-wise manner.
In this BFS approach:
Time Complexity: O(V+E).
Space Complexity: O(V).
1#include <vector>
2#include <unordered_map>
3#include <queue>
4using namespace std;
5
6class Node {
7public:
8 int val;
9 vector<Node*> neighbors;
10 Node() {
11 val = 0;
12 neighbors = vector<Node*>();
13 }
14 Node(int _val) {
15 val = _val;
16 neighbors = vector<Node*>();
17 }
18 Node(int _val, vector<Node*> _neighbors) {
19 val = _val;
20 neighbors = _neighbors;
21 }
22};
23
24class Solution {
25public:
26 Node* cloneGraph(Node* node) {
27 if (!node) return NULL;
28 unordered_map<Node*, Node*> visited;
29 queue<Node*> q;
30 q.push(node);
31 visited[node] = new Node(node->val);
32
33 while (!q.empty()) {
34 auto n = q.front(); q.pop();
35 for (auto neighbor : n->neighbors) {
36 if (visited.find(neighbor) == visited.end()) {
37 visited[neighbor] = new Node(neighbor->val);
38 q.push(neighbor);
39 }
40 visited[n]->neighbors.push_back(visited[neighbor]);
41 }
42 }
43 return visited[node];
44 }
45};
This C++ BFS-based solution utilizes a queue to explore nodes level by level. We maintain a map visited
to keep track of the original to clone node mapping. Each node and its neighbors are iteratively visited, cloned, and linked.