力扣OJ题——随机链表的复制

题目:

138. 随机链表的复制

给你一个长度为 n 的链表,每个节点包含一个额外增加的随机指针 random ,该指针可以指向链表中的任何节点或空节点。

要求:构造这个链表的 深拷贝

 深拷贝应该正好由 n 个 全新 节点组成,其中每个新节点的值都设为其对应的原节点的值。新节点的 next 指针和 random 指针也都应指向复制链表中的新节点,并使原链表和复制链表中的这些指针能够表示相同的链表状态。

注意:复制链表中的指针都不应指向原链表中的节点 

示例:

思路一:暴力求解

1.拷贝链表

2.处理拷贝链表每个节点的random指针,这里封装一个函数来实现~

通过分析,思路一的时间复杂度应该是O(N^2),这是一种很直观的思路,下面我们来实现

1.拷贝链表,代码如下:

struct Node *copy = (struct Node*)malloc(sizeof(struct Node)),*p;p = copy;struct Node *cur = head;//cur指向原链表的头节点while(cur!=NULL){struct Node *curcopy=(struct Node*)malloc(sizeof(struct Node));curcopy->val=cur->val;curcopy->random=NULL;copy->next=curcopy;//copy是curcopy的前驱,起着拷贝链表的连接作用copy = copy->next;//copy往后走cur=cur->next;//cur往后走}copy->next=NULL;//当cur = NULL后的处理

2.处理拷贝链表每个节点的random指针,代码如下:

cur=head;//让在第一步使用过的cur回到原链表的头copy=p->next;//让在第一步使用过的copy来到拷贝链表的头while(cur!=NULL){if(cur->random==NULL){copy->random=NULL;}else{//find函数是用来找到random指向的节点copy->random=find(head,p->next,cur->random);}copy=copy->next;cur=cur->next;}

find函数的实现代码如下:


//find函数找到random指向的节点//head为原链表的头,copy为拷贝链表的头,dest为目标
struct Node* find(struct Node* head,struct Node* copy,struct Node* dest)
{while(head!=dest){head=head->next;copy=copy->next;}return copy;
}

下面我们将第一步与第二步的代码整合起来,就形成完整的代码

struct Node* find(struct Node* head,struct Node* copy,struct Node* dest);
struct Node* copyRandomList(struct Node* head){if(head == NULL) return NULL;struct Node *copy = (struct Node*)malloc(sizeof(struct Node)),*p;p = copy;struct Node *cur = head;while(cur!=NULL){struct Node *curcopy=(struct Node*)malloc(sizeof(struct Node));curcopy->val=cur->val;curcopy->random=NULL;copy->next=curcopy;copy = copy->next;cur=cur->next;}copy->next=NULL;cur=head;copy=p->next;while(cur!=NULL){if(cur->random==NULL){copy->random=NULL;}else{copy->random=find(head,p->next,cur->random);}copy=copy->next;cur=cur->next;}return p->next;//返回拷贝链表的头
}//find函数找到random指向的节点
struct Node* find(struct Node* head,struct Node* copy,struct Node* dest)
{while(head!=dest){head=head->next;copy=copy->next;}return copy;
}

虽然时间复杂度为O(N^2),但力扣居然给过了

不过这里要说的是,这样暴力的思路往往不是面试官心中的答案,所以下面我们来看一下更优的思路二

思路二:

1.插入拷贝节点在原节点的后面

2.控制拷贝节点的random

   copy -> random = cur ->random->next(这句是精华)

3.解下拷贝节点,同时恢复原链表的指向

思路二的巧妙就在于它在建立拷贝链表时就和原链表有了联系,有了这种联系,之后处理random指针就能更高效,直接就能找到、、

我们看一下思路二的时间复杂度,已经变成了O(N),这就非常的nice

下面我们就来实现这个思路

第一步:插入拷贝节点在原节点的后面,代码如下

    struct Node* cur = head;//cur在原节点的头while (cur != NULL)//第一步{struct Node* copy = (struct Node*)malloc(sizeof(struct Node));copy->val = cur->val;//插入copy->next = cur->next;cur->next = copy;//继续往下cur = copy->next;}

第二步:控制拷贝节点的random,代码如下

    cur = head;//恢复cur到原节点的头while (cur != NULL) //第二步{struct Node* curcopy =  cur->next;//拷贝节点if (cur->random == NULL)  curcopy->random = NULL;elsecurcopy->random =cur->random->next;cur = curcopy->next;}

第三步:解下拷贝节点使其成为独立的链表,同时恢复原链表的指向,代码如下

     cur = head;//恢复cur到原节点的头//第三步struct Node* copyhead=NULL,*copytail=NULL;//定义拷贝链表的头和尾while (cur != NULL){struct Node* curcopy = cur->next;struct Node* curnext = curcopy->next;// curnext是原链表cur的下一个节点//通过尾插使拷贝节点成为独立的链表if(copyhead == NULL)//处理头copyhead = copytail = curcopy;else //处理其他情况{copytail->next = curcopy;copytail = copytail->next;}cur->next= curnext;//恢复原链表的指向cur = curnext;//cur往后走}

下面我们将这三步的代码整合起来,就形成完整的代码

struct Node* copyRandomList(struct Node* head){if  (head == NULL)   return NULL;struct Node* cur = head;//cur在原节点的头while (cur != NULL)//第一步{struct Node* copy = (struct Node*)malloc(sizeof(struct Node));copy->val = cur->val;//插入copy->next = cur->next;cur->next = copy;//继续往下cur = copy->next;}cur = head;//恢复cur到原节点的头while (cur != NULL) //第二步{struct Node* curcopy =  cur->next;//拷贝节点if (cur->random == NULL)  curcopy->random = NULL;elsecurcopy->random =cur->random->next;cur = curcopy->next;}cur = head;//第三步struct Node* copyhead=NULL,*copytail=NULL;while (cur != NULL){struct Node* curcopy = cur->next;struct Node* curnext = curcopy->next;// curnext是原链表cur的下一个节点if(copyhead == NULL)copyhead = copytail = curcopy;else {copytail->next = curcopy;copytail = copytail->next;}cur->next= curnext;//恢复原链表cur = curnext;}return copyhead;//返回拷贝链表的头
}

代码走起来

就过了

好啦,到此为止,今天的随机链表的复制问题就结束啦,如果文中分析,题解代码有不足的地方欢迎大家在评论区讨论和指正

让我们在接下来的时间里一起学习,一起进步吧~

c0fe1378f4b1464abb37998a472b5961.jpg

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/695619.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Vulhub 练习 DC-4靶机完整复现

1.工具 kali:攻击机 IP地址:192.168.200.4 DC-4:靶机 IP地址:暂时未知 2.注意 这里搭建环境两台机器应该选用同类的网络连接方式:这里两台的连接方式为模式 二、信息收集 1.主机发现 找寻同网段下存活的主机(可…

ChatGPT调教指南 | 咒语指南 | Prompts提示词教程(二)

在我们开始探索人工智能的世界时,了解如何与之有效沉浸交流是至关重要的。想象一下,你手中有一把钥匙,可以解锁与OpenAI的GPT模型沟通的无限可能。这把钥匙就是——正确的提示词(prompts)。无论你是AI领域的新手&#…

不要抱怨,不如抱 Java 运算符吧 (1)

本篇会加入个人的所谓‘鱼式疯言’ ❤️❤️❤️鱼式疯言:❤️❤️❤️此疯言非彼疯言 而是理解过并总结出来通俗易懂的大白话, 小编会尽可能的在每个概念后插入鱼式疯言,帮助大家理解的. 🤭🤭🤭可能说的不是那么严谨.但小编初心是能让更多人…

Windows系统中定时执行python脚本

背景:本地Windows系统指定目录下会有文件的修改新增,这些变化的文件需要定时的被上传到git仓库中,这样不需要每次变更手动上传了。 首先编写一个检测文件夹下文件变化并且上传git仓库的python脚本(确保你已经在E:\edc_workspace\data_edc_et…

一键安装ROS适用于Ubuntu22/20/18

一键安装ROS适用于Ubuntu22/20/18 1、简介 ROS(Robot Operating System,机器人操作系统)是一个用于机器人软件开发的框架。它提供了一套工具和库,用于机器人应用程序的开发、测试和部署。ROS是由美国斯坦福大学机器人实验室&…

笔试题详解(C语言进阶)

前言 欢迎阅读本篇文章!本篇文章通过一个笔试题来加强我们对C语言的理解,希望对你有帮助。后续我会写一个栏目,集合我见到的C语言题目,进行分析讲解。 1、题目一 判断下面程序的输出结果:(下面说的地址4/8字节是因为对…

Docker部署Halo容器并结合内网穿透实现公网访问本地个人博客

文章目录 1. Docker部署Halo1.1 检查Docker版本如果未安装Docker可参考已安装Docker步骤:1.2 在Docker中部署Halo 2. Linux安装Cpolar2.1 打开服务器防火墙2.2 安装cpolar内网穿透 3. 配置Halo个人博客公网地址4. 固定Halo公网地址 本文主要介绍如何在CentOS 7系统使…

JAVAEE初阶 网络编程(十一)

HTTP协议 一. GET和POST方法二. Post和Get的区别三. HTTP协议中的请求报头部分1. Host2.Content-Length3.Content-Type4.User-Agent5. Referer6.Cookie 一. GET和POST方法 Get方法应用场景有很多,比如下列这个例子. Post方法主要应用在上传和登录上. 一般来说,Get方法没有body,…

百度百科词条在网络推广中的六大作用

也许很多网友都发现了,在网上查资料,百科词条往往是优先展示的。一方面因为百科是搜索引擎自身的平台,另一方面就是因为百科信息权威,网友认可度高。所以企业开展网络营销,百科营销是一块重要阵地。 也有的企业认为百科…

Autosar-Mcal配置详解-GPT

3.3.1添加GPT模块 方法与添加Dio相似,可参加Dio模块添加方法。 3.3.2 创建、配置GPT通道 1)根据需求创建GPT通道(即创建几个定时器) 本例中创建了3个定时器通道:1ms,100us,OsTimer。 2)配置GPT通道 配置T…

Android全新UI框架之常用ComposeUI组件

在Compose中,每个组件都是一个带有Composable注解的函数,被称为Composable。Compose已经预置了很多基于MD设计规范的Composable组件。 在布局方面,Compose提供了Column、Row、Box三种布局组件(感觉跟flutter差不多),类似于传统视图…

​LeetCode解法汇总106. 从中序与后序遍历序列构造二叉树

目录链接: 力扣编程题-解法汇总_分享记录-CSDN博客 GitHub同步刷题项目: https://github.com/September26/java-algorithms 原题链接: 力扣(LeetCode)官网 - 全球极客挚爱的技术成长平台 描述: 给定两个…

如何使用ArcGIS Pro生成等高线

无论在制图还是规划中,经常会使用到等高线,大多数情况下,从网上获取的高程数据都是DEM文件,我们可以通过ArcGIS Pro来生成等高线,这里为大家介绍一下生成方法,希望能对你有所帮助。 数据来源 教程所使用的…

PHP WebSocket:技术解析与实用指南

本文旨在帮助初学者掌握在PHP中使用WebSocket的关键概念和技术。我们将深入讨论从建立连接、绑定到监听等各方面的操作,并提供易于理解和实践的指导。 一、socket协议的简介 WebSocket是什么,有什么优点 WebSocket是一个持久化的协议,这是…

洛谷 【算法1-2】排序

【算法1-2】排序 - 题单 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn) 鄙人不才,刷洛谷,迎蓝桥,【算法1-2】排序 已刷,现将 AC 代码献上,望有助于各位 P1271 选举学生会 【深基9.例1】选举学生会 - 洛谷 题目 解答…

二.西瓜书——线性模型、决策树

第三章 线性模型 1.线性回归 “线性回归”(linear regression)试图学得一个线性模型以尽可能准确地预测实值输出标记. 2.对数几率回归 假设我们认为示例所对应的输出标记是在指数尺度上变化,那就可将输出标记的对数作为线性模型逼近的目标,即 由此&…

投简历这样投就对了

相关文章 写一份让HR过目不忘的简历 后悔没有早点发现这篇嵌入式简历模板 了解校招类型 投简历这样投就对了 前言一、线上投简历需要注意的东西1.线上APP投递简历(以BOSS为例)(1)填好个人信息(2)注意打招呼…

堆的结构实现与应用

目录 前言: 1.认识堆 a.如何认识堆? b.大根堆与小根堆 c.堆应用的简单认识 2.堆的结构与要实现的功能 3.向上调整算法 4.向下调整算法 5.向堆插入数据并建堆 6.堆的大小 7.堆的判空 8.取堆顶数据 9.删除堆顶数据 10.向上调整时间复杂度 11.向下调整时…

rtsp推拉流

1.搭建视频服务器 smart-rtmpd: smart_rtmpd 是一款 rtmp、rtsp 服务器,非常好用,解压既运行,支持跨平台,无任何依赖,性能和 SRS 相比不分上下 2.推拉流 下载windows版本ffmpeg,并设置环境变量. 推流 ffmpeg -re -st…

170基于matlab的DNCNN图像降噪

基于matlab的DNCNN图像降噪,网络分为三部分,第一部分为ConvRelu(一层),第二部分为ConvBNRelu(若干层),第三部分为Conv(一层),网络层数为17或者20层…