题目描述:

输入一个复杂链表(每个节点中有节点值,以及两个指针,一个指向下一个节点,另一个特殊指针指向任意一个节点),返回结果为复制后复杂链表的head。(注意,输出结果中请不要返回参数中的节点引用,否则判题程序会直接返回空)

解题思路:

注意题目中括号里的话,直接返回输入的函数的参数,是会被判空的,还是老老实实的创建节点,做复制。

思路一:

先复制一次链表,将每个节点的next指针连接到下一节点。第二步再处理random指针。

新旧链表同步扫描,针对每个节点i,在旧链表中判断其random指针是否为空,若不为空,保存其random指针所值节点r。再同步新旧链表从头开始扫描节点,在旧链表中找到与r相等的节点j,将此时新链表i这个位置节点的random指针指向新链表中j这个位置的节点。

这个方法的需要两个循环,时间复杂度为O(n^2)。

思路二:

在第一次复制链表时,利用一个hash表,将每一个旧链表中的节点与新节点中的对应节点做一个映射。

第二遍同步扫描新旧链表时,查询当前旧链表节点的random指针指向i节点,若不为空,则将新链表对应节点的random指针指向hash表hash[i]所映射的新链表中节点j。

这个方法的通过辅助空间降低时间复杂度,时间复杂度为O(n)。

代码:

思路一:

/*
struct RandomListNode {
int label;
struct RandomListNode *next, *random;
RandomListNode(int x) :
label(x), next(NULL), random(NULL) {
}
};
*/
class Solution {
public:
RandomListNode* Clone(RandomListNode* pHead)
{
if(pHead==nullptr)
return pHead;
RandomListNode* tmp = pHead;
RandomListNode* cur = new RandomListNode(tmp->label);
RandomListNode* nhead = cur;
tmp = tmp->next;
while(tmp!=nullptr)
{
RandomListNode* n = new RandomListNode(tmp->label);
cur->next = n;
cur = cur->next;
tmp = tmp->next;
}
tmp = pHead;
cur = nhead;
RandomListNode *r, *head_of_ran, *nhead_of_ran;
while(tmp!=nullptr && cur!=nullptr)
{
if(tmp->random!=nullptr)
{
r = tmp->random;
head_of_ran = pHead;
nhead_of_ran = nhead;
while(head_of_ran != nullptr)
{
if(head_of_ran != r)
{
head_of_ran = head_of_ran->next;
nhead_of_ran = nhead_of_ran->next;
}
else
{
cur->random = nhead_of_ran;
break;
}
}
}
tmp = tmp->next;
cur = cur->next;
}
return nhead; }
};

思路二:

/*
struct RandomListNode {
int label;
struct RandomListNode *next, *random;
RandomListNode(int x) :
label(x), next(NULL), random(NULL) {
}
};
*/
class Solution {
public:
RandomListNode* Clone(RandomListNode* pHead)
{
if(pHead==nullptr)
return pHead;
map<RandomListNode*, RandomListNode*> OldtoNew;
RandomListNode* tmp = pHead;
RandomListNode* cur = new RandomListNode(tmp->label);
RandomListNode* nhead = cur;
OldtoNew[tmp] = cur;
tmp = tmp->next;
while(tmp!=nullptr)
{
RandomListNode* n = new RandomListNode(tmp->label);
cur->next = n;
cur = cur->next;
OldtoNew[tmp] = cur;
tmp = tmp->next;
}
tmp = pHead;
cur = nhead;
while(tmp!=nullptr)
{
if(tmp->random!=nullptr)
{
cur->random = OldtoNew[tmp->random];
}
tmp = tmp->next;
cur = cur->next;
}
return nhead; }
};

剑指offer:复杂链表的复制的更多相关文章

  1. 剑指Offer——复杂链表的复制

    题目描述: 输入一个复杂链表(每个节点中有节点值,以及两个指针,一个指向下一个节点,另一个特殊指针指向任意一个节点),返回结果为复制后复杂链表的head.(注意,输出结果中请不要返回参数中的节点引用, ...

  2. 剑指offer 复杂链表的复制 (有向图的复制)

    时间复杂度O(3N) 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 ...

  3. 用js刷剑指offer(复杂链表的复制)

    题目描述 输入一个复杂链表(每个节点中有节点值,以及两个指针,一个指向下一个节点,另一个特殊指针指向任意一个节点),返回结果为复制后复杂链表的head.(注意,输出结果中请不要返回参数中的节点引用,否 ...

  4. 《剑指offer》 链表中倒数第k个节点

    本题来自<剑指offer> 链表中倒数第k个节点 题目: 输入一个链表,输出该链表中倒数第k个结点. 思路: 倒数第k个节点,而且只能访问一遍链表,定义两个节点,两者之间相差k个距离,遍历 ...

  5. 剑指Offer:链表中环的入口节点【23】

    剑指Offer:链表中环的入口节点[23] 题目描述 给一个链表,若其中包含环,请找出该链表的环的入口结点,否则,输出null. 题目分析 第一步确定链表中是否包含环,怎么确定呢?我们定义两个指针橙和 ...

  6. 剑指Offer:链表中倒数第k个结点【22】

    剑指Offer:链表中倒数第k个结点[22] 题目描述 输入一个链表,输出该链表中倒数第k个结点. 解题思考 我们定义两个指针L和R,R事先移动K-1个位置,然后两者同时往后移动直到遇到R的下个节点为 ...

  7. 剑指 Offer 22. 链表中倒数第k个节点

    剑指 Offer 22. 链表中倒数第k个节点 Offer 22 常规解法 常规解法其实很容易可以想到,只需要先求出链表的长度,然后再次遍历取指定长度的链接即可. package com.walega ...

  8. 力扣 - 剑指 Offer 22. 链表中倒数第k个节点

    题目 剑指 Offer 22. 链表中倒数第k个节点 思路1(栈) 既然要倒数第k个节点,那我们直接把所有节点放到栈(先进后出)里面,然后pop弹出k个元素就可以了 代码 class Solution ...

  9. 【剑指Offer】链表中倒数第k个节点 解题报告(Python)

    [剑指Offer]链表中倒数第k个节点 解题报告(Python) 标签(空格分隔): LeetCode 题目地址:https://www.nowcoder.com/ta/coding-intervie ...

  10. 【剑指Offer】链表中环的入口结点 解题报告(Python)

    [剑指Offer]链表中环的入口结点 解题报告(Python) 标签(空格分隔): 剑指Offer 题目地址:https://www.nowcoder.com/ta/coding-interviews ...

随机推荐

  1. 软工实践第二次作业-sudoku

    说明 Github项目地址 作业题目 解题思路 一开始拿到的时候,有一个思路,可以先填写全盘的"1",然后在插空填满全盘的"2".后来觉得自己理不清那个思路.遂 ...

  2. 玩转FusionCharts:Y轴数字形式(如去掉K)

    玩转FusionCharts:Y轴数字形式(如去掉K) 如果运行FusionCharts带的例子,你会发现FusionCharts表中的数字(通常是Y轴)会带上’k’,也就是如20000,会变成20k ...

  3. 用JQuery操作元素的style属性

    可以直接利用css()方法获取元素的样式属性,JQuery代码如下: 1 $("p").css("color");  //获取p元素的样式颜色 无论color属 ...

  4. 【Java多线程】线程状态、线程池状态

    线程状态: 线程共包括以下5种状态.1. 新建状态(New) 线程对象被创建后,就进入了新建状态.例如,Thread thread = new Thread().2. 就绪状态(Runnable) 也 ...

  5. day14 Python函数

    函数def,严格来讲有个return返回值 过程就是没有return返回值的函数 #过程 def test01(): msg = 'liuhaoran' print(msg) #函数 def test ...

  6. MATLAB——神经网络pureline激活函数

  7. Objective-C 类簇深入理解

    类簇(class cluster),是一种基于抽象工厂的设计模式,广泛运用于系统的Foundation框架.顾名思义,即一坨类,这里指的是继承自同一父类的一组私有子类.这种实现既可以简化公共接口,又保 ...

  8. Luogu P1525 关押罪犯

    传送门 首先 这是一个并查集= = 这道题其实明白了还挺简单的qwq 思路: 因为只看仇恨值最大的一对儿,所以把他们从大到小排序,越大的就尽量分开,直到不能再分为止qwq q[x]表示x最大的敌人(x ...

  9. VS2017上执行VS2013项目错误MSB802之解决方案

    进行想把我编写的数字图像处理软件MagicHouse更新到最新的VS2017开发环境下,原来的开发环境是VS2013.但是用VS2017打开项目并编译时,系统报错误MSB802,如下图所示. 其实Vi ...

  10. Identity(五)

    本文摘自 ASP.NET MVC 随想录—— 使用ASP.NET Identity实现基于声明的授权,高级篇 在这篇文章中,我将继续ASP.NET Identity 之旅,这也是ASP.NET Ide ...