forked from terrytong0876/LintCode-1
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathClone Graph.java
executable file
·146 lines (117 loc) · 4.95 KB
/
Clone Graph.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
M
Use HashMap to mark cloned nodes.
先能复制多少Node复制多少。然后把neighbor 加上
```
/*
Clone an undirected graph. Each node in the graph contains a label and a list of its neighbors.
OJ's undirected graph serialization:
Nodes are labeled uniquely.
We use # as a separator for each node, and , as a separator for node label and each neighbor of the node.
As an example, consider the serialized graph {0,1,2#1,2#2,2}.
The graph has a total of three nodes, and therefore contains three parts as separated by #.
First node is labeled as 0. Connect node 0 to both nodes 1 and 2.
Second node is labeled as 1. Connect node 1 to node 2.
Third node is labeled as 2. Connect node 2 to node 2 (itself), thus forming a self-cycle.
Visually, the graph looks like the following:
1
/ \
/ \
0 --- 2
/ \
\_/
Hide Tags Depth-first Search Breadth-first Search Graph
*/
/*
//NEED TO RUN THIS ON LINT
Thoughts: 12.12.2015
The original thoughs of using ArrayList, and using a index to track of which node has not been visited.
It's alright, but it uses extra space, and basically copie all nodes again.
It's similar to using a queue.
At the end, it's doing O(m * n)
Maybe can improve this.
Need a queue and process each element. and a hashmap to track duplicates.
1. make sure the node is no duplicate
2. make sure to all all child
border: case: node == nul, or node has not child, return a new instance of it'self?
*/
public class Solution {
public UndirectedGraphNode cloneGraph(UndirectedGraphNode node) {
if (node == null || node.neighbors.size() == 0) {
return node;
}
HashMap<UndirectedGraphNode, UndirectedGraphNode> map =
new HashMap<UndirectedGraphNode, UndirectedGraphNode>();
Queue<UndirectedGraphNode> queue = new LinkedList<UndirectedGraphNode>();
queue.offer(node);
//process each node
while (!queue.isEmpty()) {
UndirectedGraphNode curr = queue.poll();
UndirectedGraphNode newNode;
if (!map.containsKey(curr)) {
map.put(curr, new UndirectedGraphNode(curr.label));
}
UndirectedGraphNode newNode = map.get(curr);
//Add neighbors for each node
for (UndirectedGraphNode neighbor : curr.neighbors) {
UndirectedGraphNode newNeighbor;
if (!map.containsKey(neighbor)) {
map.put(neighbor, new UndirectedGraphNode(neighbor.label));
}
newNeighbor = map.get(neighbor);
newNode.neighbors.add(newNeighbor);
}//end for
}//end while
return map.get(node);
}
}
/*
Thinking process:
1. Clone all nodes available: using HashMap to go through all possible query. No duplicates added using HashMap.
HashMap map has the list of all new nodes. No neighbors added yet
<key,value> = <original node, new node with just a label (without neighbor list)>
At same time, the arrayList nodes has all original nodes(with neighbors) in Breadth-first order.
2. Add neighbor for nodes in map:
- Locate the 'newNode' from map by using the key: the original node
- loop through the original node's neighbor size
- use original neighbor as key to get the new neighbor instance from map
- add this new neighbor instance to the neighbor list of 'newNode'
*/
/**
* Definition for undirected graph.
* class UndirectedGraphNode {
* int label;
* List<UndirectedGraphNode> neighbors;
* UndirectedGraphNode(int x) { label = x; neighbors = new ArrayList<UndirectedGraphNode>(); }
* };
*/
public class Solution {
public UndirectedGraphNode cloneGraph(UndirectedGraphNode node) {
if (node == null) {
return node;
}
ArrayList<UndirectedGraphNode> nodes = new ArrayList<UndirectedGraphNode>();
nodes.add(node);
HashMap<UndirectedGraphNode, UndirectedGraphNode> map = new HashMap<UndirectedGraphNode, UndirectedGraphNode>();
map.put(node, new UndirectedGraphNode(node.label));
int start = 0;
//Clone nodes without neighbors:
while (start < nodes.size()) {
List<UndirectedGraphNode> neighbors = nodes.get(start++).neighbors;
for (int i = 0; i < neighbors.size(); i++) {
if (!map.containsKey(neighbors.get(i))) {
map.put(neighbors.get(i), new UndirectedGraphNode(neighbors.get(i).label));
nodes.add(neighbors.get(i));
}
}
}
// Clone neighbors:
for (int i = 0; i < nodes.size(); i++) {
UndirectedGraphNode newNode = map.get(nodes.get(i));
for (int j = 0; j < nodes.get(i).neighbors.size(); j++) {
newNode.neighbors.add(map.get(nodes.get(i).neighbors.get(j)));
}
}
return map.get(node);
}
}
```