「这是我参与11月更文挑战的第18天,活动详情查看:2021最后一次更文挑战」
复制带随机指针的链表
天下傻逼独一个就是我,我忘记了选c语言,用c++结果错误看一脸懵逼
image-20211028093751124
题目
image-20211027185429832
这题链表的复制难的地方就是随机指针random 如何复制他的指向,很多人的确想到了malloc节点,我也想到了malloc一个一个的节点,但是基本所有人都卡死在这里了,就是我们如何链,在这里我们就会又遇到先天性大佬的天赋思维力的压制,上一题是空间联想的压制。(来吧我小码农就喜欢给先天以及后天的大佬锤炼)
思想
普通拷贝
image-20211027202915998
实际上想到这样也已经了不起了,有点磨具雏形了
顺藤摸瓜(这个是真的奇兵记,暗度陈仓的感觉)
image-20211028004315304
脱裤子还需提裤子人(上面暗度陈仓,这个就是单刀直入)
image-20211028004358166
代码实现
cur为NULL的时候就是停止copy的时候
struct Node* cur = head;
if(!cur)
return NULL;
while(cur)
{
//每次我们都要malloc一个copy节点
struct Node* copy = (struct Node*)malloc(sizeof(struct Node));
//开始给数据
copy->val = cur->val;
//开始插入
copy->next = cur->next;
cur->next = copy;
//开始cur移动
cur = copy->next;
}
复制代码
有人说为什么不在上面malloc节点的时候就链,因为那时候小孩还没出生你就让你的还在上学吗,没错为了卷你们,我准备偷偷生个小孩
//节点copy好后开始random相链
cur = head;
while(cur)
{
//重新把之前开辟的节点给copy维护
struct Node* copy = cur->next;
if(cur->random == NULL)
copy->random = NULL;
else
copy->random = cur->random->next;//上面的GIF的核心代码
cur = copy->next;
}
复制代码
然后拿下来一个个尾插,就是单刀直入你要还原原来的链表顺序
//开始一个一个尾插起来
struct Node* copyHead = NULL;
struct Node* copyTail = NULL;
cur = head;
while(cur)
{
//重新把之前开辟的节点给copy维护
struct Node* copy = cur->next;
//这边你需要一个next来还原原来链表顺序,你不能用完人家对人家不负责任
struct Node* next = copy->next;
if(copyHead == NULL)
{
copyHead = copy;
copyTail = copy;
}
else
{
copyTail->next = copy;
copyTail = copy;
}
cur->next = next;//这一步看你对原来链表负不负责任
cur = next;
}
return copyHead;
复制代码
image-20211028094601284
struct Node* copyRandomList(struct Node* head) {
struct Node* cur = head;
if(!cur)
return NULL;
while(cur)
{
//每次我们都要malloc一个copy节点
struct Node* copy = (struct Node*)malloc(sizeof(struct Node));
//开始给数据
copy->val = cur->val;
//开始插入
copy->next = cur->next;
cur->next = copy;
//开始cur移动
cur = copy->next;
}
//节点copy好后开始random相链
cur = head;
while(cur)
{
//重新把之前开辟的节点给copy维护
struct Node* copy = cur->next;
if(cur->random == NULL)
copy->random = NULL;
else
copy->random = cur->random->next;//上面的GIF的核心代码
cur = copy->next;
}
//开始一个一个尾插起来
struct Node* copyHead = NULL;
struct Node* copyTail = NULL;
cur = head;
while(cur)
{
//重新把之前开辟的节点给copy维护
struct Node* copy = cur->next;
//这边你需要一个next来还原原来链表顺序,你不能用完人家对人家不负责任
struct Node* next = copy->next;
if(copyHead == NULL)
{
copyHead = copy;
copyTail = copy;
}
else
{
copyTail->next = copy;
copyTail = copy;
}
cur->next = next;//这一步看你对原来链表负不负责任
cur = next;
}
return copyHead;
}
复制代码