《剑指 Offer (第 2 版)》第 35 题:复杂链表的复制

第 35 题:复杂链表的复制

传送门:复杂链表的复刻牛客网 online judge 地址

请实现一个函数可以复制一个复杂链表。

在复杂链表中,每个结点除了有一个指针指向下一个结点外,还有一个额外的指针指向链表中的任意结点或者 null 。

分析:一些细节要考虑到,特别是空结点的判断,以下两种方法都要遍历链表一遍以上,即遍历链表一遍是不够的。

思路1:有点巧妙,穿针引线,新旧断开。

Python 代码:

class ListNode(object):
    def __init__(self, x):
        self.val = x
        self.next = None
        self.random = None


class Solution(object):
    def copyRandomList(self, head):
        """
        :type head: ListNode
        :rtype: ListNode
        """
        if head is None:
            return None

        # 第 1 步:根据 next 指针复制出一个新旧合一的链表
        cur_node = head
        while cur_node:
            # 先暂存 cur_node 的 next_node 下一次遍历要用
            next_node = cur_node.next
            
            new_node = ListNode(cur_node.val)

            cur_node.next = new_node
            new_node.next = next_node
            # 指针遍历到下一个结点
            cur_node = next_node
        # 第 2 步:根据旧结点 random 指针,给新结点的 random 指针做出正确的指向
        cur_node = head
        while cur_node:
            new_node = cur_node.next
            # 同样要先暂存 cur_node.next_node
            next_node = new_node.next
            # 要记得做非空判断,因为题目中说了 random 有可能为空
            new_node.random = cur_node.random.next if cur_node.random else None
            cur_node = next_node

        # 第 3 步:旧结点和新结点分离(拆分链表)
        cur_node = head
        res = cur_node.next
        while cur_node:
            new_node = cur_node.next
            # 同样要先暂存下一个结点
            next_node = new_node.next
            # 恢复原始结点
            cur_node.next = new_node.next
            # 恢复拷贝结点
            # 这里也要同样注意空指针的问题,克隆结点的最后一个结点的下一个结点是 null
            new_node.next = next_node.next if next_node else None
            cur_node = next_node
        return res

Java 代码:

class RandomListNode {
    int label;
    RandomListNode next = null;
    RandomListNode random = null;

    RandomListNode(int label) {
        this.label = label;
    }
}

public class Solution {
    public RandomListNode Clone(RandomListNode pHead) {
        // 极端情况,一定要写先出来
        if (pHead == null) {
            return null;
        }

        RandomListNode curNode = pHead;
        // 第 1 步:根据 next 指针复制出一个新旧合一的链表
        // 此时,奇数索引(从 1 开始计算)的结点是旧的结点,偶数索引的结点是新的结点
        RandomListNode nextNode;
        RandomListNode copyNode;
        while (curNode != null) {
            nextNode = curNode.next;
            copyNode = new RandomListNode(curNode.label);
            curNode.next = copyNode;
            copyNode.next = nextNode;
            curNode = nextNode;
        }

        // 第 2 步:根据旧结点 random 指针,给新结点的 random 指针做出正确的指向
        // 指针复位到起始结点
        curNode = pHead;
        while (curNode != null) {
            copyNode = curNode.next;
            nextNode = copyNode.next;
            // 特别注意
            // 特别注意
            // 特别注意:有的结点很可能 random 的指向为空(题目中明确说明)
            // 所以:只要遇到 next 引用的时候,一定要注意判断是否为空
            copyNode.random = curNode.random == null ? null : curNode.random.next;
            curNode = nextNode;
        }

        // 第 3 步:旧结点和新结点分离(拆分链表)
        curNode = pHead;
        RandomListNode res = pHead.next;
        while (curNode != null) {
            copyNode = curNode.next;
            nextNode = copyNode.next;
            // 恢复原始结点
            curNode.next = copyNode.next;
            // 恢复拷贝结点
            // 这里也要同样注意空指针的问题,克隆结点的最后一个结点的下一个结点是 null
            copyNode.next = copyNode.next == null ? null : copyNode.next.next;
            curNode = nextNode;
        }
        return res;
    }
}

C++ 写法:

《剑指 Offer (第 2 版)》第 35 题:复杂链表的复制

Java 版本里面还有一个网友的写法。

思路2:使用哈希表,复制出随机访问的指针。

Python 代码:

class ListNode(object):
    def __init__(self, x):
        self.val = x
        self.next = None
        self.random = None


class Solution(object):
    def copyRandomList(self, head):
        """
        :type head: ListNode
        :rtype: ListNode
        """
        if head is None:
            return None
        map = dict()
        dummy_node = ListNode(-1)
        # p 指针用于新链表的 next 指针赋值
        p = dummy_node

        # 遍历一次链表,做两件事
        # 1、复制结点
        # 2、只管 next 指针
        cur_node = head
        while cur_node:
            new_node = ListNode(cur_node.val)
            # 把新旧结点的对应关系放在一个 map 里
            map[cur_node] = new_node
            cur_node = cur_node.next

            p.next = new_node
            p = new_node

        # 接下来做随机指针的复制
        for old_node, new_node in map.items():
            # 要记得判断是否为空,否则 None 不能作为 map  key
            if old_node.random:
                new_node.random = map[old_node.random]
        return dummy_node.next

Java 代码:

public class Solution3 {
    public RandomListNode Clone(RandomListNode pHead) {
        Map<RandomListNode, RandomListNode> map = new HashMap<>();
        RandomListNode curNode = pHead;
        // 体会这里设置虚拟头结点的必要性
        RandomListNode dummyNode = new RandomListNode(-1);
        RandomListNode p = dummyNode;
        // 完成复制 next 结点,并且将对应关系放入 Hash 表
        while (curNode != null) {
            RandomListNode newNode = new RandomListNode(curNode.label);
            map.put(curNode, newNode);
            curNode = curNode.next;
            p.next = newNode;
            p = newNode;
        }
        // 完成复制链表的 random 指针的赋值
        Set<Map.Entry<RandomListNode, RandomListNode>> entrySet = map.entrySet();
        for (Map.Entry<RandomListNode, RandomListNode> entry : entrySet) {
            entry.getValue().random = map.get(entry.getKey().random);
        }
        return dummyNode.next;
    }
}

Java 代码:特别注意:在复制“指针”的时候,对空对象的判定,画图是十分关键的。

class RandomListNode {
    int label;
    RandomListNode next = null;
    RandomListNode random = null;

    RandomListNode(int label) {
        this.label = label;
    }
}

public class Solution {
    public RandomListNode Clone(RandomListNode pHead) {
        // 极端情况,一定要写先出来
        if (pHead == null) {
            return null;
        }
        
        RandomListNode curNode = pHead;
        // 第 1 步:根据 next 指针复制出一个新旧合一的链表
        // 此时,奇数索引(从 1 开始计算)的结点是旧的结点,偶数索引的结点是新的结点
        RandomListNode nextNode;
        RandomListNode copyNode;
        while (curNode != null) {
            nextNode = curNode.next;
            copyNode = new RandomListNode(curNode.label);
            curNode.next = copyNode;
            copyNode.next = nextNode;
            curNode = nextNode;
        }

        // 第 2 步:根据旧结点 random 指针,给新结点的 random 指针做出正确的指向
        // 指针复位到起始结点
        curNode = pHead;
        while (curNode != null) {
            copyNode = curNode.next;
            nextNode = copyNode.next;
            // 特别注意
            // 特别注意
            // 特别注意:有的结点很可能 random 的指向为空(题目中明确说明)
            // 所以:只要遇到 next 引用的时候,一定要注意判断是否为空
            copyNode.random = curNode.random == null ? null : curNode.random.next;
            curNode = nextNode;
        }

        // 第 3 步:旧结点和新结点分离(拆分链表)
        curNode = pHead;
        RandomListNode res = pHead.next;
        while (curNode != null) {
            copyNode = curNode.next;
            nextNode = copyNode.next;
            // 恢复原始结点
            curNode.next = copyNode.next;
            // 恢复拷贝结点
            // 这里也要同样注意空指针的问题,克隆结点的最后一个结点的下一个结点是 null
            copyNode.next = copyNode.next == null ? null : copyNode.next.next;
            curNode = nextNode;
        }
        return res;
    }
}

Java 代码:

import java.util.HashMap;
import java.util.Map;
import java.util.Set;

/**
 * 使用 Hash 表的方式
 *
 * @author liwei
 */
public class Solution2 {
    public RandomListNode Clone(RandomListNode pHead) {
        Map<RandomListNode, RandomListNode> map = new HashMap<>();
        RandomListNode curNode = pHead;
        // 体会这里设置虚拟头结点的必要性
        RandomListNode dummyNode = new RandomListNode(-1);
        RandomListNode p = dummyNode;
        // 完成复制 next 结点,并且将对应关系放入 Hash 表
        while (curNode != null) {
            RandomListNode newNode = new RandomListNode(curNode.label);
            map.put(curNode, newNode);
            curNode = curNode.next;
            p.next = newNode;
            p = newNode;
        }
        // 完成复制链表的 random 指针的赋值
        Set<Map.Entry<RandomListNode, RandomListNode>> entrySet = map.entrySet();
        for (Map.Entry<RandomListNode, RandomListNode> entry : entrySet) {
            entry.getValue().random = map.get(entry.getKey().random);
        }
        return dummyNode.next;
    }
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 223,689评论 6 521
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 95,685评论 3 401
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 170,676评论 0 366
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 60,496评论 1 300
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 69,512评论 6 399
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 53,044评论 1 314
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 41,423评论 3 427
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 40,399评论 0 278
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 46,923评论 1 323
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,973评论 3 343
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 41,117评论 1 354
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 36,761评论 5 350
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 42,440评论 3 336
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,929评论 0 25
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 34,045评论 1 275
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 49,609评论 3 380
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 46,147评论 2 363

推荐阅读更多精彩内容