[LeetCode] 133. Clone Graph

Given a reference of a node in a connected undirected graph.

Return a deep copy (clone) of the graph.

Each node in the graph contains a value (int) and a list (List[Node]) of its neighbors.

class Node {
    public int val;
    public List<Node> neighbors;
}

Test case format:

For simplicity, each node's value is the same as the node's index (1-indexed). For example, the first node with val == 1, the second node with val == 2, and so on. The graph is represented in the test case using an adjacency list.

An adjacency list is a collection of unordered lists used to represent a finite graph. Each list describes the set of neighbors of a node in the graph.

The given node will always be the first node with val = 1. You must return the copy of the given node as a reference to the cloned graph.

Input: adjList = [[2,4],[1,3],[2,4],[1,3]]
Output: [[2,4],[1,3],[2,4],[1,3]]
Explanation: There are 4 nodes in the graph.
1st node (val = 1)'s neighbors are 2nd node (val = 2) and 4th node (val = 4).
2nd node (val = 2)'s neighbors are 1st node (val = 1) and 3rd node (val = 3).
3rd node (val = 3)'s neighbors are 2nd node (val = 2) and 4th node (val = 4).
4th node (val = 4)'s neighbors are 1st node (val = 1) and 3rd node (val = 3).

克隆图。

给你无向 连通 图中一个节点的引用,请你返回该图的 深拷贝(克隆)。

图中的每个节点都包含它的值 val(int) 和其邻居的列表(list[Node])。

class Node {
    public int val;
    public List<Node> neighbors;
}

测试用例格式:

简单起见,每个节点的值都和它的索引相同。例如,第一个节点值为 1(val = 1),第二个节点值为 2(val = 2),以此类推。该图在测试用例中使用邻接列表表示。

邻接列表 是用于表示有限图的无序列表的集合。每个列表都描述了图中节点的邻居集。

给定节点将始终是图中的第一个节点(值为 1)。你必须将 给定节点的拷贝 作为对克隆图的引用返回。

来源:力扣(LeetCode)
链接:https://leetcode.cn/problems/clone-graph
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

题目要求对一个图做深度复制。这里需要被复制的内容不光是图上的每个节点,同时也需要为每个节点的邻接点做深度复制。

既然是克隆/复制一个数据结构,一定涉及到两点,一是遍历,二是记录。图的遍历无非就是 BFS 或者 DFS,记录基本都会用到 hashmap,除了链表的复制也许可以不需要用到 hashmap,因为可以把复制出来的节点加在原节点之后。参见138题

这个题也可以用 DFS 做但是我个人觉得 BFS 比较好记。具体的做法是创建一个visited hashmap记住<原来的node,新的node>之间的对应关系,和一个 queue。

一开始,先把当前唯一的节点 node 加入 queue 并且把当前节点当做 key 存入 hashmap,map 的 value 是一个有着相同 node.val 但是只有一个空的邻接表(neighbors)的node;

然后从 queue 中开始弹出元素,每弹出一个元素 cur,判断这个 node 的每一个邻居是否存在于 hashmap,将不存在的邻居加入 hashmap,同时也将这些刚刚加入 hashmap 中的邻居再次加入 queue 进行下一轮遍历;如果有某一个邻居已经存在于 hashmap 了,则往 cur 的邻接表里面添加这个已经存在的邻居的 copy。

BFS

时间O(n) - 遍历了所有节点

空间O(n) - queue + hashmap

Java实现

/*
class Node {
    public int val;
    public List<Node> neighbors;
    
    public Node() {
        val = 0;
        neighbors = new ArrayList<Node>();
    }
    
    public Node(int _val) {
        val = _val;
        neighbors = new ArrayList<Node>();
    }
    
    public Node(int _val, ArrayList<Node> _neighbors) {
        val = _val;
        neighbors = _neighbors;
    }
}
*/

class Solution {
    public Node cloneGraph(Node node) {
        // corner case
        if (node == null) {
            return null;
        }
        // normal case
        HashMap<Node, Node> visited = new HashMap<>();
        Queue<Node> queue = new LinkedList<>();
        queue.add(node);
        // <oldNode, newNode>
        visited.put(node, new Node(node.val, new ArrayList<>()));
        while (!queue.isEmpty()) {
            Node cur = queue.poll();
            for (Node neighbor : cur.neighbors) {
                if (!visited.containsKey(neighbor)) {
                    visited.put(neighbor, new Node(neighbor.val, new ArrayList()));
                    queue.offer(neighbor);
                }
                // add a copy of this neighbor to the copy of cur
                visited.get(cur).neighbors.add(visited.get(neighbor));
            }
        }
        return visited.get(node);
    }
}

 

DFS

时间O(n)

空间O(n)

Java实现

 1 class Solution {
 2     public Node cloneGraph(Node node) {
 3         HashMap<Node, Node> map = new HashMap<>();
 4         return dfs(node, map);
 5     }
 6 
 7     private Node dfs(Node node, HashMap<Node, Node> map) {
 8         if (node == null) {
 9             return null;
10         }
11         if (map.containsKey(node)) {
12             return map.get(node);
13         }
14         Node clone = new Node(node.val, new ArrayList<>());
15         map.put(node, clone);
16         for (Node neighbor : node.neighbors) {
17             clone.neighbors.add(dfs(neighbor, map));
18         }
19         return clone;
20     }
21 }

 

相关题目

133. Clone Graph

138. Copy List with Random Pointer

1485. Clone Binary Tree With Random Pointer

1490. Clone N-ary Tree

LeetCode 题目总结

posted @ 2020-05-03 11:45  CNoodle  阅读(557)  评论(0编辑  收藏  举报