【Leetcode Top 100】138. 随机链表的复制

问题背景

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

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

例如,如果原链表中有 X X X 和 Y Y Y 两个节点,其中 X . r a n d o m = Y X.random = Y X.random=Y。那么在复制链表中对应的两个节点 x x x 和 y y y,同样有 x . r a n d o m = y x.random = y x.random=y。

返回复制链表的头节点。

数据约束

  • 0 ≤ n ≤ 1000 0 \le n \le 1000 0≤n≤1000
  • − 1 0 4 ≤ N o d e . v a l ≤ 1 0 4 -10 ^ 4 \le Node.val \le 10 ^ 4 −104≤Node.val≤104
  • N o d e . r a n d o m Node.random Node.random 为 n u l l null null 或指向链表中的节点。

解题过程

经典链表操作题,解决的关键在于能否想到把新建的复制节点添加到原节点的后一个。

通过上述方案复制完整个链表之后,只要能够分离链表即可,参考 奇偶链表,本题中由于原链表的后一个节点是它本身的复制,一定存在,分离的时候可以少一个判断。

注意数据范围,头节点可能为空,要单独处理。

具体实现

java 复制代码
/*
// Definition for a Node.
class Node {
    int val;
    Node next;
    Node random;

    public Node(int val) {
        this.val = val;
        this.next = null;
        this.random = null;
    }
}
*/

class Solution {
    public Node copyRandomList(Node head) {
        // 特判头节点为空的情形
        if(head == null) {
            return null;
        }
        // 在原链表的每个节点之后新建复制节点
        Node cur = head, next;
        while(cur != null) {
            next = cur.next;
            cur.next = new Node(cur.val, cur.next); // 题中没说明这个构造器,实际上是存在的
            cur.next.next = next;
            cur = cur.next.next;
        }
        // 重置工作指针
        cur = head;
        // 给每个新节点的随机域赋值
        while(cur != null) {
            if(cur.random != null) {
                cur.next.random = cur.random.next;
            }
            cur = cur.next.next;
        }
        // 分离原链表和复制之后的链表
        Node copyHead = head.next, copy;
        cur = head;
        while(cur.next.next != null) {
            copy = cur.next; // 记录下一个节点
            cur.next = cur.next.next; // 调整原链表的 next 指针
            copy.next = copy.next.next; // 调整新链表的 next 指针
            cur = cur.next; // 后移工作指针
        }
        // 原链表的最后一个节点要指空
        cur.next = null;
        return copyHead;
    }
}
相关推荐
SuperBear12 分钟前
轮转数组 循环换位解法
算法
用户403159863966315 分钟前
版本号升级统计
java·算法
DoraBigHead38 分钟前
小哆啦解题记 · 射箭引爆气球,怎么射最省箭?
算法
AI_Keymaker1 小时前
手术台上的AlphaGo:约翰霍普金斯大学发布自主手术机器人
算法·机器人
AI_Keymaker1 小时前
人类早期驯服野生机器人珍贵影像(不是)
算法·机器人
今禾1 小时前
一行代码引发的血案:new Array(5) 到底发生了什么?
前端·javascript·算法
橙几2 小时前
击败了90%的解法?Two Sum 从 O(n²) 到 O(n) 的优化之路
算法
叶子爱分享2 小时前
经典排序算法之归并排序(Merge Sort)
算法·排序算法
珹洺2 小时前
C++算法竞赛篇:DevC++ 如何进行debug调试
java·c++·算法
Norvyn_72 小时前
LeetCode|Day18|20. 有效的括号|Python刷题笔记
笔记·python·leetcode