138. 随机链表的复制 --力扣 --JAVA

题目

给你一个长度为 n 的链表,每个节点包含一个额外增加的随机指针 random ,该指针可以指向链表中的任何节点或空节点。

构造这个链表的 深拷贝 。 深拷贝应该正好由 n全新 节点组成,其中每个新节点的值都设为其对应的原节点的值。新节点的 next 指针和 random 指针也都应指向复制链表中的新节点,并使原链表和复制链表中的这些指针能够表示相同的链表状态。复制链表中的指针都不应指向原链表中的节点

例如,如果原链表中有 XY 两个节点,其中 X.random --> Y 。那么在复制链表中对应的两个节点 xy ,同样有 x.random --> y

返回复制链表的头节点。

用一个由 n 个节点组成的链表来表示输入/输出中的链表。每个节点用一个 [val, random_index] 表示:

  • val:一个表示 Node.val 的整数。
  • random_index:随机指针指向的节点索引(范围从 0n-1);如果不指向任何节点,则为 null

你的代码 接受原链表的头节点 head 作为传入参数。

解题思路

  1. 先排除特殊情况:若链表为null则直接返回null;
  2. 使用Map存储原始链表的节点和对应索引;
  3. 使用List来存储复制后链表的节点;
  4. while循环完成节点复制、next链接和存储;
  5. 从Map和List中获取首节点进行第二次遍历;
  6. 第二次遍历建立random链接。

代码展示

java 复制代码
class Solution {
    public Node copyRandomList(Node head) {
        //排除特殊情况
        if (head == null){
            return null;
        }
        Node ans = new Node(head.val);
        //需要获取random的索引,所以用map比遍历List要便捷
        Map<Node, Integer> headMap = new HashMap<>();
        //只需要根据索引获取对应节点,List本身是有序的,所以不需要用Map
        List<Node> ansList = new ArrayList<>();
        int index = 0;
        headMap.put(head, index);
        ansList.add(ans);
        //遍历生成所有节点并存储起来
        while (head.next != null){
            head = head.next;
            ans.next = new Node(head.val);
            index++;
            headMap.put(head, index);
            ansList.add(ans.next);
            ans = ans.next;
        }
        for (Node node : headMap.keySet()){
            if(headMap.get(node) == 0){
                head = node;
                break;
            }
        }
        ans = ansList.get(0);
        while (head != null && ans != null){
            Integer num = headMap.get(head.random);
            Node temp = null;
            if(num != null){
                temp = ansList.get(num);
            }
            ans.random = temp;
            head = head.next;
            ans = ans.next;
        }
        return ansList.get(0);
    }
}
相关推荐
lingling0097 分钟前
机械臂动作捕捉系统选型指南:从需求到方案,NOKOV 度量光学动捕成优选
人工智能·算法
吃着火锅x唱着歌17 分钟前
LeetCode 410.分割数组的最大值
数据结构·算法·leetcode
Benny_Tang39 分钟前
题解:P7989 [USACO21DEC] Bracelet Crossings G
c++·算法
YSRM42 分钟前
Leetcode+Java+图论+最小生成树&拓扑排序
java·leetcode·图论
YSRM43 分钟前
Leetcode+Java+图论+并查集
算法·leetcode·图论
小白杨树树1 小时前
【C++】力扣hot100错误总结
c++·leetcode·c#
康谋自动驾驶2 小时前
拆解3D Gaussian Splatting:原理框架、实战 demo 与自驾仿真落地探索!
算法·数学建模·3d·自动驾驶·汽车
violet-lz3 小时前
数据结构八大排序:希尔排序-原理解析+C语言实现+优化+面试题
数据结构·算法·排序算法
ezl1fe4 小时前
第一篇:把任意 HTTP API 一键变成 Agent 工具
人工智能·后端·算法