题目描述:

输入一个复杂链表(每个节点中有节点值,以及两个指针,一个指向下一个节点,另一个特殊指针指向任意一个节点),返回结果为复制后复杂链表的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. ABAP性能和优化

    哪些工具可以用于性能优化? ST05-性能追踪.包含SQL追踪加RFC,队列和缓存追踪.SQL追踪主要用于测量程序中select语句的性能. SE30-运行时分析.用于测量应用的性能. SAT是过时的 ...

  2. Go学习笔记02-基本语法

    目录 变量定义 代码示例 变量定义 内建变量类型 复数回顾 强制类型转换 常量 枚举类型 变量定义 代码示例 package main import "fmt" /* 此处定义的变 ...

  3. Django基础之urls

    一  Django简介 二   视图层之应用系统 一  Django简介 Django:   urls:路径与视图函数的映射关系   views:逻辑处理   models:与数据库相关的操作   t ...

  4. Activity总结

    1)Activity在app构建中的位置: 2)与window和视图体系.事件处理的关系: 3)生命周期: 4)启动方式: 5)activity管理: 6)加载管理:

  5. exec dbms_stats.gather_schema_stats 手动优化统计

    Oracle10g或以上版本.exec dbms_stats.gather_schema_stats(ownname => 'DFMS', options => 'GATHER AUTO' ...

  6. meta标签使用方法总结

    meta标签是用来描述一个HTML网页文档的属性,比如该网页的作者,日期,网页的关键字,刷新,网页等级设定等等,是文档中的最基本的元信息,本文是对meta标签的学习总结,内容大多来自网上各个地方,非原 ...

  7. win7(64位)Sql server 用T-sql读取本地数据文件dbf的数据文件

    原文地址:https://www.cnblogs.com/cl1006/p/9924066.html 第一步启用Ad Hoc Distributed Queries  在SQLserver执行以下的语 ...

  8. 获取键盘的ascii码

    waitKey(1) & 0xFF获取当前按的键的ascii码,

  9. ubuntu 系统判断优盘的指定文件存在

    有很多的时候 会出现没有用的优盘路径 如果代码中写的是绝对路径 就有可能读不到优盘的内容 ,以下代码就是可以解决这样的问题 我已经封装成一个类 upanpath.h #ifndef UPANPATH_ ...

  10. Zephyr的Time、Timer、sleep

    正如Linux下一样,关于时间的系统函数可以分为三类:时间值.睡眠一段时间以及延迟执行. 在Zephyr上对应是什么样子呢?带着这个疑问,去了解一下这些函数. 以及他们与suspend之间的关系? 是 ...