剑指 Offer 35. 复杂链表的复制
难度中等
请实现 copyRandomList 函数,复制一个复杂链表。在复杂链表中,每个节点除了有一个 next 指针指向下一个节点,还有一个 random 指针指向链表中的任意节点或者 null。
示例 1:
![](https://img-blog.csdnimg.cn/img_convert/53456352aaf4d2aee46596c28f6247d6.png)
输入:head = [[7,null],[13,0],[11,4],[10,2],[1,0]]
输出:[[7,null],[13,0],[11,4],[10,2],[1,0]]
示例 2:
![](https://img-blog.csdnimg.cn/img_convert/0f7fb1e2ebd66cbbe9a59a1d9a277585.png)
输入:head = [[1,1],[2,1]]
输出:[[1,1],[2,1]]
示例 3:
![](https://img-blog.csdnimg.cn/img_convert/f61ada1cfcf5569551c918cd1f1f97a8.png)
输入:head = [[3,null],[3,0],[3,null]]
输出:[[3,null],[3,0],[3,null]]
示例 4:
输入:head = []
输出:[]
解释:给定的链表为空(空指针),因此返回 null。
提示:
-10000 <= Node.val <= 10000
Node.random 为空(null)或指向链表中的节点。
节点数目不超过 1000 。
注意:本题与主站 138 题相同:https://leetcode-cn.com/problems/copy-list-with-random-pointer/
解题思路:
// 原地修改解法
1.复制一个新的节点在原有节点之后,如 1 -> 2 -> 3 -> null 复制完就是 1 -> 1 -> 2 -> 2 -> 3 - > 3 -> null
2.从头开始遍历链表,通过 cur.next.random = cur.random.next 可以将复制节点的随机指针串起来,当然需要判断 cur.random 是否存在
3.将复制完的链表一分为二 根据以上信息,
图解:
![](https://img-blog.csdnimg.cn/img_convert/912a049b2f9f674bac1b9bf07b7ab24d.png)
代码:
class Node {
int val;
Node next;
Node random;
public Node(int val) {
this.val = val;
this.next = null;
this.random = null;
}
}
/*剑指 Offer 35. 复杂链表的复制
请实现 copyRandomList 函数,复制一个复杂链表。在复杂链表中,每个节点除了有一个 next
指针指向下一个节点,还有一个 random 指针指向链表中的任意节点或者 null。
示例 1:
输入:head = [[7,null],[13,0],[11,4],[10,2],[1,0]]
输出:[[7,null],[13,0],[11,4],[10,2],[1,0]]*/
public class Demo35 {
// 剑指 Offer 35. 复杂链表的复制
/*1.复制一个新的节点在原有节点之后,如 1 -> 2 -> 3 -> null 复制完就是 1 -> 1 -> 2 -> 2 -> 3 - > 3 -> null
2.从头开始遍历链表,通过 cur.next.random = cur.random.next 可以将复制节点的随机指针串起来,当然需要判断 cur.random 是否存在
3.将复制完的链表一分为二 根据以上信息,我们不难写出以下代码*/
// 原地修改解法
public Node copyRandomList(Node head) {
if(head==null) return head;//判断链表为空时
// 复制链表
Node cur = head;
while (cur!=null){
Node copyNode = new Node(cur.val);
copyNode.next=cur.next;
cur.next=copyNode;
cur=cur.next.next;
}
// 复制随机节点
cur = head;
while (cur!=null){
if (cur.random!=null){
if (cur.random!=null)
cur.next.random=cur.random.next;
}
cur=cur.next.next;
}
// 将复制后链表一分为二
cur=head;
// 链表
Node curcopy=head.next;
// 作为指针使用
Node copyHead = head.next;
while (cur!=null){
cur.next=cur.next.next;
cur=cur.next;
if (curcopy.next!=null){
curcopy.next=curcopy.next.next;
curcopy = curcopy.next;
}
}
return copyHead;
}