138. Copy List with Random Pointer

目录

题目描述

方法一、使用哈希表

方法二、不使用哈希表


题目描述

问题的关键是,random指针指向的是原链表的结点,这个原链表的结点对应哪一个新链表的结点呢?有两种办法。一是用哈希表。另一种是复制原链表的每一个结点,并将新结点接在原结点的后面组成一个长度加倍的链表,这样原结点的直接后继就是该原结点对应的新结点。

方法一、使用哈希表

unordered_map<Node*,Node*> new_table;//键是原链表中的旧结点,值是新链表中的新结点

unordered_map<Node*,Node*> random_table;//键是原链表中的旧结点,值是该旧结点的random指针指向的结点

cpp 复制代码
/*
// Definition for a Node.
class Node {
public:
    int val;
    Node* next;
    Node* random;
    
    Node(int _val) {
        val = _val;
        next = NULL;
        random = NULL;
    }
};
*/

class Solution {
public:
    Node* copyRandomList(Node* head) {
        if(head == nullptr)
            return nullptr;
        unordered_map<Node*,Node*> new_table;//键是原链表中的旧结点,值是新链表中的新结点
        unordered_map<Node*,Node*> random_table;//键是原链表中的旧结点,值是该旧结点的random指针指向的结点
        Node* newHead = nullptr;
        Node* newTail = nullptr;
        Node* cur = head;

        //第一步,建立新链表
        while(cur){
            Node* node = new Node(cur->val);
            new_table.insert({cur,node});
            random_table.insert({cur,cur->random});
            if(newHead == nullptr){
                newHead = node;
                newTail = node;
            }else{
                newTail->next = node;
                newTail = node;
            }
            cur = cur->next;
        }

        //第二步,设置新链表结点的random指针
        for(const auto& [old_node,new_node] : new_table)
        {
            if(random_table[old_node]){
                new_node->random = new_table[random_table[old_node]];
            }
        }
        return newHead;
    }
};

实际上前面做法中的random_table是可以不需要的。

cpp 复制代码
/*
// Definition for a Node.
class Node {
public:
    int val;
    Node* next;
    Node* random;
    
    Node(int _val) {
        val = _val;
        next = NULL;
        random = NULL;
    }
};
*/

class Solution {
public:
    Node* copyRandomList(Node* head) {
        if(head == nullptr)
            return nullptr;
        unordered_map<Node*,Node*> new_table;//键是原链表中的旧结点,值是新链表中的新结点
        Node* newHead = nullptr;
        Node* newTail = nullptr;
        Node* cur = head;
        //第一步,建立新链表
        while(cur){
            Node* node = new Node(cur->val);
            new_table.insert({cur,node});
            if(newHead == nullptr){
                newHead = node;
                newTail = node;
            }else{
                newTail->next = node;
                newTail = node;
            }
            cur = cur->next;
        }

        cur = head;
        Node* new_cur = newHead;
        //第二步,设置新链表结点的random指针
        while(cur){
            if(cur->random){
                new_cur->random = new_table[cur->random];
            }
            cur = cur->next;
            new_cur = new_cur->next;
        }

        return newHead;
    }
};

方法二、不使用哈希表

第一步,复制每一个旧结点,并将新结点接在旧结点的后面组成新旧结点交替出现的长度翻倍的大链表。

第二步,设置新结点的random指针。

第三步,从大链表中提取出新结点组成新链表,并将原链表复原。

需要着重强调的是,第二步和第三步必须分开来做。

cpp 复制代码
/*
// Definition for a Node.
class Node {
public:
    int val;
    Node* next;
    Node* random;
    
    Node(int _val) {
        val = _val;
        next = NULL;
        random = NULL;
    }
};
*/

class Solution {
public:
    Node* copyRandomList(Node* head) {
        if(head == nullptr)
            return nullptr;
        Node* cur = head;
        //复制每一个旧结点,并将新结点接在旧结点的后面
        while(cur){
            Node* node = new Node(cur->val);
            Node* temp = cur->next;
            node->next = cur->next;
            cur->next = node;
            cur = temp;
        }

        cur = head;
        //设置新结点的random指针
        while(cur){
            if(cur->random)
                cur->next->random = cur->random->next;
            cur = cur->next->next;//原链表向前走一位
        }

        Node* new_head = nullptr;
        Node* new_tail = nullptr;
        cur = head;
        //提取出新结点组成新链表,并将原链表复原
        while(cur){
            if(new_head == nullptr){
                new_head = cur->next;
                new_tail = cur->next;
            }else{
                new_tail->next = cur->next;
                new_tail = cur->next;
            }
            cur->next = cur->next->next;//复原原链表
            cur = cur->next;//原链表向前走一步
        }

        return new_head;
    }
};
相关推荐
qeen876 小时前
【数据结构】建堆的时间复杂度讨论与TOP-K问题
c语言·数据结构·c++·学习·
图码7 小时前
如何用多种方法判断字符串是否为回文?
开发语言·数据结构·c++·算法·阿里云·线性回归·数字雕刻
我星期八休息7 小时前
IT疑难杂症诊疗室:AI时代工程师Superpowers进化论
linux·开发语言·数据结构·人工智能·python·散列表
漂流瓶jz7 小时前
UVA-1152 和为0的4个值 题解答案代码 算法竞赛入门经典第二版
数据结构·算法·二分查找·题解·aoapc·算法竞赛入门经典·uva
你撅嘴真丑8 小时前
map 与 set容器的应用--话题焦点人物
数据结构
生成论实验室8 小时前
《事件关系阴阳博弈动力学:识势应势之道》第二篇:阴阳博弈——认知的动力学基础
数据结构·人工智能·科技·神经网络·算法
li1670902708 小时前
第二十七章:智能指针
c语言·数据结构·c++·visual studio
风筝在晴天搁浅9 小时前
LeetCode 92.反转链表Ⅱ
算法·leetcode·链表
WL_Aurora10 小时前
Python 算法基础篇之链表
python·算法·链表
代码中介商11 小时前
数据结构开篇:从问题到解决方案
数据结构