【链表面试题】——剑指 Offer : 复杂链表(带随机指针)的复制

news2025/1/16 5:05:06

文章目录

    • 前言
    • 1.题目介绍
    • 2. 题目分析
    • 3. 思路讲解
      • 思路1
      • 思路2
        • 步骤1
        • 步骤2
        • 步骤3
    • 4. 分析图及源码展示

前言

这篇文章,我们一起来解决一道与链表相关的经典面试题:复杂链表(带随机指针)的复制。
在这里插入图片描述

1.题目介绍

我们先来一起了解一下这道题:

这道题是《剑指offer》上的一道经典题目:
在这里插入图片描述
在力扣上也有原题:
链接: link
这篇文章,就给大家详细讲解一下这道题。

我们一起来看一下题目:

在这里插入图片描述

题目呢,看起来还挺长的。

但是我们不能上去被题目就吓到了,其实这个题目就是让我们复制链表嘛,给我们一个链表,我们要自己再创建一个和它一样的链表就行了嘛。

🆗,那接下来,我们就来一起分析一下这道题。

2. 题目分析

那既然这道题是让我们复制链表的,那我们就先来思考一下应该如何复制?

通过前面的学习,我们已经学会了如果创建一个链表,那复制的话,就是创建一个一模一样的链表嘛。

我们就拿一个题目给出的输入样例来分析一下:

在这里插入图片描述

那要复制这样一个链表,是不是好像也不难啊。

它有5个结点,我们就创建的链表也带5个结点就行了嘛,每个结点1个数据域,2个指针域,next指针依次存下一个结点的地址。
对吧,这些操作好像都不难。

那求解这道题的关键之处或者说难点在哪呢?

是不是就麻烦在这个随机指针的问题啊,这是不是比较难搞啊。
每个节点包含一个额外增加的随机指针 random ,该指针可以指向链表中的任何节点或空节点。
我们除了要把链表的连接关系复制出来,每个结点的随机指针指向哪里,我们也要复制出来的。
但是每个结点的随机指针的指向随机的,可能指向空,或者是任意一个结点,那我们要复制随机指针,就必须知道每个结点的随机指针的指向,这就不好搞了。

我们看是很容易看出来的,但是如何让我们的程序面对不同的输入,都能正确找到每一个结点随机指针的指向,并进行复制呢?

3. 思路讲解

思路1

首先思路1就是暴力求解:

复制随机指针的时候,每个复制结点的random指针,我们都要一一去寻找它对应的源结点指向的是第几个结点(如果指向空是比较好搞的),然后让复制结点也指向对应的结点,但是注意不能看它指向的数值是几,因为不同结点数据域的数值可能是一样的。
在这里插入图片描述
但是这种解法时间复杂度是O(N^2) ,不是太好,我们就不实现了。

接下来我们再讲另外一种比较优的算法。

思路2

这个思路是什么呢?需要我们做三件事情:

步骤1

  1. 创建拷贝结点链接到源链表每一个结点的后面

在这里插入图片描述
那这么做有什么目的呢?
这里先不解释,等到下一步操作时我们就知道为什么要这么做了。

那这一步代码要如何实现呢?

那也很简单,循环对链表进行遍历,每次循环都创建一个copy结点,copy结点的数据域的值和源节点相同,然后连接到源结点后面,一次向后直到遍历结束。

	struct Node* cur=head;
    //拷贝结点到源节点后面
    while(cur)
    {
        struct Node* copy=(struct Node*)malloc(sizeof(struct Node));
        copy->val=cur->val;
        //链接
        copy->next=cur->next;
        cur->next=copy;
        //cur往后走
        cur=copy->next;
    }

步骤2

接下来是第二步:

  1. 设置拷贝结点的random指针域

在这里插入图片描述

这么设置呢?

如果源结点的random域为空,则拷贝结点的random域也直接置为空;如果不为空,拷贝结点的random域指向 【其对应源节点的random域指向的源节点】的next对应的拷贝结点。
在这里插入图片描述
大家可以结合图理解一下这一步的操作。
这样拷贝结点的random指针设置起来是不是就很方便了,这也是我们第一步把拷贝结点链接在源节点后面的原因。

    cur=head;
    //设置拷贝结点的random域
    while(cur)
    {
        struct Node* copy=cur->next;
        if(cur->random==NULL)
        {
            copy->random=NULL;
        }
        else
        {
            copy->random=cur->random->next;
        }
        cur=copy->next;
    }

步骤3

那接下来就是第3步,也是最后一步了。

  1. 第三步:将拷贝结点解绑下来,链接组成最终要返回的拷贝链表

经过前面两步的努力,拷贝链表的所有结点都已经存在了,而且它们的随机指针random也设置好了,那现在我们把所有的拷贝结点从源链表上解下来,再组成一个完整的链表不就完成了吗?当然因为前两步的操作我们改变了原链表,所有最后最好将它还原一下,避免题目测试的时候会检查我们是否原链表被改变了。
在这里插入图片描述

那要将所有结点组成链表,我们就可以依次尾插。

这里给不给头结点都可以,我们这里选择不给头结点,不过这样第一个结点尾插进行一次判断就行了。

    //将拷贝结点解下尾插成新链表
    cur=head;
    struct Node* copyhead=NULL;
    struct Node* copytail=NULL;
    while(cur)
    {
        struct Node* copy=cur->next;
        //将原链表还原
        cur->next=copy->next;
        //尾插
        if(copytail==NULL)
        {
            copyhead=copytail=copy;
        }
        else
        {
            copytail->next=copy;
            copytail=copy;
        }
        //cur向后走
        cur=copy->next;
    }

拷贝链表创建好,最好作为函数返回值返回就行了。

4. 分析图及源码展示

在这里插入图片描述

源码:

struct Node* copyRandomList(struct Node* head) {
	struct Node* cur=head;
    //拷贝结点到源节点后面
    while(cur)
    {
        struct Node* copy=(struct Node*)malloc(sizeof(struct Node));
        copy->val=cur->val;
        //链接
        copy->next=cur->next;
        cur->next=copy;
        //cur往后走
        cur=copy->next;
    }
    cur=head;
    //设置拷贝结点的random域
    while(cur)
    {
        struct Node* copy=cur->next;
        if(cur->random==NULL)
        {
            copy->random=NULL;
        }
        else
        {
            copy->random=cur->random->next;
        }
        cur=copy->next;
    }
    //将拷贝结点解下尾插成新链表
    cur=head;
    struct Node* copyhead=NULL;
    struct Node* copytail=NULL;
    while(cur)
    {
        struct Node* copy=cur->next;
        //将原链表还原
        cur->next=copy->next;
        //尾插
        if(copytail==NULL)
        {
            copyhead=copytail=copy;
        }
        else
        {
            copytail->next=copy;
            copytail=copy;
        }
        //cur向后走
        cur=copy->next;
    }
    return copyhead;
}

🆗,那这道题的讲解就完了,希望能帮助到大家。
如果有写的不好的地方,欢迎大家指正!!!
在这里插入图片描述

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

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

相关文章

Java的继承到底是怎么回事?看这篇让你明明白白

一. 引言 在学习面向对象后,我们就可以使用类来描述对象共有的特征(属性)和行为举止(方法),如果我们用类来描述猫、狗和企鹅,可以进行如下编码: public class Cat {private String name;//名字private int age;//年龄private St…

操作系统,计算机网络,数据库刷题笔记11

操作系统,计算机网络,数据库刷题笔记11 2022找工作是学历、能力和运气的超强结合体,遇到寒冬,大厂不招人,可能很多算法学生都得去找开发,测开 测开的话,你就得学数据库,sql&#xf…

kubelet源码分析 syncLoopIteration(一) configCh

kubelet源码分析 syncLoopIteration syncLoopIteration里有四个chan管道。分别是configCh、plegCh、syncCh、housekeepingCh。这篇主要聊一下这四个管道的由来。 一、configCh configCh是通过list&watch的API SERVER获得的数据。然后在本地进行比对,推送到c…

Qt-Web混合开发-QtWebChannel实现Qt与Web通信交互-进阶功能(6)

Qt-Web混合开发-QtWebChannel实现Qt与Web通信交互-进阶功能🥬 文章目录Qt-Web混合开发-QtWebChannel实现Qt与Web通信交互-进阶功能🥬1、概述🌽2、实现效果🍆3、实现功能🍒4、关键代码🥝5、源代码&#x1f9…

Android基础学习(二十二)—— View的事件分发(1)

一、View的层级关系 二、View的事件分发机制 1、MotionEvent ——点击事件 点击事件用MotionEvent来表示 ACTION_DOWN:手指刚接触屏幕 ACTION_MOVE:手指在屏幕上移动 ACTION_UP:手指从屏幕上松开的一瞬间 点击事件的事件分发&#xff0…

OM6621系列国产M4F内核低功耗BLE5.1大内存SoC蓝牙芯片

目录OM6621系列简介OM6621P系列芯片特性应用领域OM6621系列简介 随着5G与物联网时代的到来,智慧城市、电动出行、智能家居、可穿戴设备等应用高速发展,低功耗蓝牙技术在近几年智能化浪潮中的地位也尤为重要。OM6621系列的开发即是为解决国内低功耗蓝牙应…

Linux安装docker 保姆级教程

一、docker介绍 Docker 是 2014 年最为火爆的技术之一,几乎所有的程序员都听说过它。Docker 是一种“轻量级”容器技术,它几乎动摇了传统虚拟化技术的地位,现在国内外已经有越来越多的公司开始逐步使用 Docker 来替换现有的虚拟化平台了。 二…

图为科技深圳人工智能产业协会重磅推出边缘计算机全新概念

人工智能作为提升区域竞争力的重要战略,全国各地都在推动发展,人工智能是未来科技创新发展的风向标,也是产业变革升级的关键驱动力,我国在《“十四五”数字经济发展规划》及《工业互联网创新发展行动计划(2021-2023年)》中&#x…

Linux基础(4)-进程管理

该文章主要为完成实训任务,详细实现过程及结果见【参考文章】 参考文章:https://howard2005.blog.csdn.net/article/details/127066383?spm1001.2014.3001.5502 文章目录一、查看进程1. 进程查看命令 - ps2. Liunx进程状态3. 观察进程变化命令 - top4. …

b站黑马的Vue快速入门案例代码——【axios+Vue】天知道(天气信息查询功能)

目录 目标效果: 更换的新接口接口文档: 天知道新的Get请求接口:http://ajax-api.itheima.net/api/weather html文件中注意因为接口更换,要修改原代码为如下红字部分: 重点原理: (1)v-on可以…

环形链表问题

文章目录环形链表问题1.环形链表题干思路延申问题总结2. 环形链表 II题干思路环形链表问题 环形链表就是一个链表没有结束的位置,链表的最后一个节点它会指向链表中的某一个节点形成一个环。 拿力扣的两到题目来看 1.环形链表 题干 给你一个链表的头节点 head …

JavaScript JSON解析

最近在uniapp中遇到了一个bug,排查后是json解析的问题。对uniapp开发比较熟悉的,应该会知道uni.navigateTo 这个API方法。这是官方提供用于跳转页面的方法。 有时候我们在跳转页面时会想传递一些参数,通常采用这样的方式 navigateTo(url, r…

oauth2.0--基础--6.1--SSO的实现原理

oauth2.0–基础–6.1–SSO的实现原理 1、什么是SSO 1.1、概念 在一个 多系统共存 的环境下,用户在一处登录后,就不用在其他系统中登录,就可以访问其他系统的资源。用户环境 浏览器:只能同一个浏览器,不会出现A浏览器…

zabbix部署【各模块超详细】

目录 安装zabbix 部署zabbix 配置zabbix 1. 修改语言 2. 监控linux端 3. 修改中文乱码 报警功能 报警音报警 邮件报警 脚本报警 邮件通知内容 图形模块 创建图形 创建聚合图形 percona mysql模板 nginx模板 克隆主机 网络发现 自动注册 主被动模式 🍁如果对你有帮助…

Handsontable复制列标题内容的功能

Handsontable复制列标题内容的功能 添加了通过使用3个新的上下文菜单选项复制列标题内容的功能:“使用标题复制”、“使用组标题复制”和“仅复制标题”。 添加了4个用于以编程方式复制列标题的新API方法:“copyCellsOnly()”、“copyWithColumnHeaders(…

vscode jupyter配置远程服务器开发

背景说明:本地vscode中使用jupyter编写本地python代码很方便,各种快捷键用的飞起。但是要做线上大数据分析时。在集群环境中搭建一个jupyter。使用网页端编写程序非常不习惯,所以想到能不能将线上的jupyter接口开出来,使用vscode远…

js-有关时间

https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Date 有关Date 返回格式 Sun Oct 10 2021 00:00:00 GMT0800 (中国标准时间) new Date() 无参数 获取当前时间new Date(value) 传入时间戳 传入一个时间戳 一个 Unix 时间戳(U…

JVM运行流程/运行时数据区

JVM运行流程 程序在执行之前先要把 java代码 转换成 字节码文件 (.class文件), JVM 首先需要把字节码通过一定的方式 类加载器 (ClassLoader) 把文件加载到内存中 运行时数据区 (Runtime Data Area) , 而字节码文件是 JVM 的一套指令集规范, 并不能直接交给底层操作系统去执行…

【大数据技术Hadoop+Spark】Hive数据仓库架构、优缺点、数据模型介绍(图文解释 超详细)

一、Hive简介 Hive起源于Facebook,Facebook公司有着大量的日志数据,而Hadoop是实现了MapReduce模式开源的分布式并行计算的框架,可轻松处理大规模数据。然而MapReduce程序对熟悉Java语言的工程师来说容易开发,但对于其他语言使用…

Anaconda为虚拟环境安装第三方库与Spyder等软件的方法

本文介绍在Anaconda中,为Python的虚拟环境安装第三方库与Spyder等配套软件的方法。 在文章Anaconda中Python虚拟环境的创建、使用与删除(https://blog.csdn.net/zhebushibiaoshifu/article/details/128334614)中,我们介绍了在Anac…