LeetCode 面试经典 150_图_克隆图(90_133_C++_中等)(深度优先:DFS)

LeetCode 面试经典 150_图_克隆图(90_133_C++_中等)

题目描述:

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

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

class Node {

public int val;

public List neighbors;

}

测试用例格式

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

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

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

输入输出样例:

示例 1:

输入 :adjList = [[2,4],[1,3],[2,4],[1,3]]
输出 :[[2,4],[1,3],[2,4],[1,3]]
解释

图中有 4 个节点。

节点 1 的值是 1,它有两个邻居:节点 2 和 4 。

节点 2 的值是 2,它有两个邻居:节点 1 和 3 。

节点 3 的值是 3,它有两个邻居:节点 2 和 4 。

节点 4 的值是 4,它有两个邻居:节点 1 和 3 。

示例 2:

输入 :adjList = [[]]
输出 :[[]]
解释:输入包含一个空列表。该图仅仅只有一个值为 1 的节点,它没有任何邻居。

示例 3:
输入 :adjList = []
输出 :[]
解释:这个图是空的,它不含任何节点。

提示:

这张图中的节点数在 [0, 100] 之间。

1 <= Node.val <= 100

每个节点值 Node.val 都是唯一的,

图中没有重复的边,也没有自环。

图是连通图,你可以从给定节点访问到所有节点。

题解:

解题思路:

思路一(深度优先:DFS):

1、深拷贝 克隆出来一个图,需要重新开辟内存空间 来存储原图值的信息。可采用 DFS 遍历原图,将遍历到的结点进行创建和连接(遍历一个结点创建一个克隆结点,在递归返回时进行连接)。需要注意存在环的处理 ,环的处理可采用记录遍历结点 来解决。当一个克隆结点创建过后,需将原结点记录一下,下次再遍历此节点时不再创建(查找创建过的结点使用哈希表)。

2、复杂度分析:

① 时间复杂度:O(N+E),图中有 N 个节点和 E 条边,:对于图中的每个节点,我们会访问一次,并且递归地遍历它的每个邻居。

② 空间复杂度:O(N),使用一个 unordered_map 来存储已经克隆的节点,最多会存储 N 个节点,因此哈希表的空间复杂度为 O(N)。

代码实现

代码实现(思路一(深度优先:DFS)):
cpp 复制代码
class Solution {
private:
    // 记录已经克隆过的节点,避免重复克隆
    unordered_map<Node*, Node*> visited;

public:
    
    // 克隆图的主函数
    Node* cloneGraph(Node* node) {
        // 如果当前节点为空,返回空指针
        if (node == nullptr) return nullptr;

        // 如果当前节点已经被克隆过,直接返回对应的克隆节点(防止在有环或互相连接的图中产生无限递归)
        if (visited.count(node)) {
            return visited[node];
        }

        // 创建一个新的节点,它是当前节点的克隆
        Node* cloneNode = new Node(node->val);

        // 将克隆节点存入 visited 哈希表,以当前节点为键,克隆节点为值
        visited[node] = cloneNode;

        // 遍历当前节点的邻居
        for (auto &neighbor : node->neighbors) {
            // 递归地克隆邻居,并将克隆的邻居添加到当前克隆节点的邻接列表中
            cloneNode->neighbors.emplace_back(cloneGraph(neighbor));
        }
        
        // 返回克隆后的节点
        return cloneNode;
    }
};

LeetCode 面试经典 150_图_克隆图(90_133)原题链接

欢迎大家和我沟通交流(✿◠‿◠)

相关推荐
苦藤新鸡17 小时前
8.最长的无重复字符的子串
c++·力扣
千金裘换酒17 小时前
LeetCode反转链表
算法·leetcode·链表
꧁Q༒ོγ꧂18 小时前
C++ 入门完全指南(四)--函数与模块化编程
开发语言·c++
汉克老师18 小时前
GESP2025年12月认证C++八级真题与解析(判断题8-10)
c++·快速排序··lcs·gesp八级·gesp8级
qq_4335545418 小时前
C++ manacher(求解回文串问题)
开发语言·c++·算法
程序员小寒18 小时前
从一道前端面试题,谈 JS 对象存储特点和运算符执行顺序
开发语言·前端·javascript·面试
圣保罗的大教堂19 小时前
leetcode 1161. 最大层内元素和 中等
leetcode
闲看云起19 小时前
LeetCode-day6:接雨水
算法·leetcode·职场和发展
HL_风神19 小时前
设计原则之迪米特
c++·学习·设计模式
黛色正浓19 小时前
leetCode-热题100-贪心合集(JavaScript)
javascript·算法·leetcode