链表经典OJ题(链表回文结构,链表带环,链表的深拷贝)

目录

前言

1.反转一个单链表。 

2.   给定一个带有头结点 head 的非空单链表,返回链表的中间结点。

3.链表的回文结构。

 4.链表带环问题(*****)

4.1是否带环

4.2 入环的节点

5.随机链表的复制(链表的深拷贝)


前言

        前面我们学习了链表,现在我们来手撕几道经典链表OJ题目吧!!!


1.反转一个单链表。 

题目链接206. 反转链表 - 力扣(LeetCode)

题解:

        在这一题,我们定义了三个指针变量,首先让prev指向NULL,prev的作用是保存cur的前面的一个节点,next是保存cur后面的节点。

        每一次循环迭代,都让cur指向前面的节点,也就是指向prev;

        再让prev去到cur的位置,cur去到next位置,最后再让next去到cur->next的位置,这样就完成了一次循环迭代。直到cur为NULL;
 

struct ListNode* reverseList(struct ListNode* head) {
    struct ListNode*prev = NULL;
    struct ListNode*cur = head;
    struct ListNode*next =head;
    while(cur)
    {
        next = cur->next;
        cur->next = prev;
        prev =cur;
        cur = next;
    }
    return prev;
}

        这样就通过了!

      


2.   给定一个带有头结点 head 的非空单链表,返回链表的中间结点。

题目链接:876. 链表的中间结点 - 力扣(LeetCode)


 

题解:

        这题思想其实非常简单,既然让返回中间节点,那么我们就定义两个指针,fast和slow,让fast走两步,slow走一步,这样fast走完全程之后,slow就只走了一半。

        需要注意的是,节点总数为奇数时,fast走到最后一个节点就结束,而节点总数为偶数时,fast节点就要走到NULL。因此结束条件就要写成(fast&&fast->next)。

struct ListNode* middleNode(struct ListNode* head){
    struct ListNode* slow = head,*fast = head;
    while(fast&&fast->next)
    {
        slow = slow->next;
        fast = fast->next->next;
    }

    return slow;
}

3.链表的回文结构。

题目链接:链表的回文结构_牛客题霸_牛客网 (nowcoder.com)


 题解:

        这一题看似复杂,实际不难。我们只需要找到中间节点,然后从中间节点反转链表 ,再分别从两个头开始遍历,若每个节点都相等,则为回文结构。

        如果是奇数个,两条链表节点数量会不会不想等呢?

        不会!因为A链表的最后一个并没有与B链表的最后一个节点断开,所以两链表有一个 公共节点,因此在奇数个节点下,两链表节点个数也相同,结束条件显而易见为:(head&&rhead)。

class PalindromeList {
public:
struct ListNode* reverseList(struct ListNode* head) {//反转链表
    struct ListNode*prv = NULL;
    struct ListNode*cur = head;
    struct ListNode*n =head;
    while(n)
    {
        n = cur->next;
        cur->next = prv;
        prv =cur;
        cur = n;
    }
    return prv;
}
struct ListNode* middleNode(struct ListNode* head){//找中间节点
    struct ListNode* slow = head,*fast = head;
    while(fast&&fast->next)
    {
        slow = slow->next;
        fast = fast->next->next;
    }

    return slow;
}
    bool chkPalindrome(ListNode* A) {
        ListNode* mid = middleNode(A);
        ListNode*rhead = reverseList(mid);
        while(A&&rhead)
        {
            if(A->val!=rhead->val)
                return false;
            A=A->next;
            rhead=rhead->next;
        }
        return true;
    }    
};


 4.链表带环问题(*****)

        链表带环问题是链表中非常重要的一类问题,在企业面试中也会经常遇到。此类问题有两种,第一种是判断链表是否带环,第二种是判断链表开始入环的节点是哪个。


4.1是否带环

题目链接:141. 环形链表 - 力扣(LeetCode)

题解:我们不禁思索,该如何判断链表是否带环呢?

        在这里我们可以定义两个指针fast和slow,让fast先走,如果到最后还能与slow相遇,那就证明该链表是带环的。在此就引出了一个问题,fast要比slow快多少?

        1.slow一次走一步fast一次走两步一定会相遇吗?

        答案是一定会相遇,因为fast和slow的距离每次都缩减1,到最后一定会减到0。

        2.slow一次走一步fast一次走三步一定会相遇吗?

        答案是不一定,如果N为偶数,fast和slow的距离每次都缩减2,最后一定会减到0。如果为奇数,最后会减到-1,这样就又开始新一轮的追击了,而且永远不会相遇。

        3.slow一次走n步fast一次走m步一定会相遇吗?(m>n>1)

        最后我们得到2L = n*C-N最后一定得到的是一个奇数,所以最后一定能追上。

bool hasCycle(struct ListNode *head) {
    struct ListNode* fast = head,*slow =head;
    while(fast&&fast->next)
    {
        fast =fast->next->next;
        slow =slow ->next;
        if(fast ==slow)
            return fast;
    }
    return NULL;
}

4.2 入环的节点

题目链接:142. 环形链表 II - 力扣(LeetCode) 、

题解:这一题与上一题是不一样的,这一题是要找到入环的节点,从哪个节点开始入环,这题难度相较于上一题显然是上升了。

        假设,fast和slow在meet点相遇,那么slow就走了(L+X)的距离,fast就走了(L+X+n*C)的距离,最后由图上等式可得,L=n*C-X,那也就是说,如果相同速度的指针,一个从相遇点开始走,另一个从入口点开始走,他们到入口与点的距离是一样的,所以一定会在入口点相遇。

        因此本题思路就出来了:1.找到相遇点 2.让相同速度的指针,一个从相遇点开始走,另一个从入口点开始走。最终就一定会相遇。

struct ListNode *detectCycle(struct ListNode *head) {
    struct ListNode *slow = head,*fast =head;
    while(fast&&fast->next)
    {
        slow = slow->next;
        fast= fast->next->next;
        if(slow == fast)
        {
            struct ListNode* meet= slow;
            while(head!=meet)
            {
                head = head->next;
                meet =meet->next;

            }
            return meet;
        }
    }
    return NULL;
}

5.随机链表的复制(链表的深拷贝)

题目链接:138. 随机链表的复制 - 力扣(LeetCode)

题解:链表的深拷贝,是链表中较难的问题,但如果把思路理清,问题也就迎刃而解了。

        我们的第一思路可能就是将每个节点的信息先拷贝下来,然后我们来处理random的指向的时候,可能就是用嵌套循环将每个节点的random进行比较,来确定指向,这样一来就造成时间复杂度到达O(N^2)了,这就不符合题目要求了。

        那么我这里就提供一种可行的思路:

                1.拷贝的节点都插入在原来节点的后面(如图)

                2.处理random的指向

        我们可以清楚的看到,原节点的下一个节点就是我们所拷贝的节点,那么让本来指向原节点random,指向原节点的下一个是不是就可以让拷贝节点的random完成正确的指向,这就解决了然random指向的问题了。如图(我们已经得到所有的拷贝节点了。)

                3.copy的节点一个一个的解下来尾插就可以得到我们想要的深拷贝出来的链表了。

完整代码:

truct 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->next->next;
    }
    cur =head;
    while(cur)
    {
        struct Node*copy= cur->next;
        if(cur->random ==NULL)
            copy->random =NULL;
        else
            copy->random = cur->random->next;
        cur = cur->next->next; 
    }
    struct Node* newhead = NULL,*fail=NULL;
    cur =head;
    while(cur)
    {
        struct Node*copy = cur->next;
        struct Node*next =copy->next;
        if(newhead==NULL)
        {
            newhead = fail = copy;
        }
        else
        {
            fail->next = copy;
            fail=fail->next;
        }
            
        cur->next = next;
        cur = next;
    }
    return newhead;
}

这里都是链表比较经典的题目,希望对你有所帮助!!!

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

会员题-力扣408-有效单词缩写

有效单词缩写 字符串可以用 缩写 进行表示,缩写 的方法是将任意数量的 不相邻 的子字符串替换为相应子串的长度。例如,字符串 “substitution” 可以缩写为(不止这几种方法): “s10n” (“s ubstitutio n”) “sub4…

numpy 基础使用

NumPy是Python中科学计算的基础包。它是一个Python库,提供多维数组对象,各种派生对象(如掩码数组和矩阵),以及用于数组快速操作的各种API,有包括数学、逻辑、形状操作、排序、选择、输入输出、离散傅立叶变…

2.OpenResty系列之Lua入门

1. Lua简介 Lua是一种轻量级的、高效的脚本编程语言,最初由巴西里约热内卢天主教大学的一个研究小组开发和发布。Lua的设计目标是提供一个简单、可嵌入、可扩展的脚本语言,官方实现完全采用 ANSI C 编写,能以 C 程序库的形式嵌入到其他应用程…

【数据结构初阶】顺序表SeqList

描述 顺序表我们可以把它想象成在一个表格里面填数据,并对数据做调整; 那我们的第一个问题是:怎么样在创建出足够的空间呢? 我们可以去堆上申请,用一个指针指向一块空间,如果申请的空间不够,我…

第十六届山东省职业院校技能大赛高职组“软件测试”赛项规程

第十六届山东省职业院校技能大赛 高职组“软件测试”赛项规程 一、赛项名称 赛项名称:软件测试 赛项组别:高职组 赛项专业大类:电子与信息大类 二、竞赛目的 软件是新一代信息技术的灵魂,是数字经济发展的基础,是…

汽车ECU的虚拟化技术初探(一)

目录 1.为什么要提汽车ECU的虚拟化? 2.虚拟化技术分类 2.1 硬件虚拟化 2.2 操作系统虚拟化 问题引入: Hypervisor是如何来管理和隔离硬件资源,保证各个不同功能的应用程序的资源使用安全和资源调度?没有MMU就做不了虚拟化&am…

Clickhouse学习笔记(11)—— 数据一致性

使用合并树引擎时,无论是ReplacingMergeTree还是SummingMergeTree,都只能保证数据的最终一致性,因为数据的去重、聚合等操作会在数据合并的期间进行,而合并会在后台以一个不确定的时间进行,因此无法预先计划&#xff1…

基于SSM的停车场管理系统设计与实现

项目描述 临近学期结束,还是毕业设计,你还在做java程序网络编程,期末作业,老师的作业要求觉得大了吗?不知道毕业设计该怎么办?网页功能的数量是否太多?没有合适的类型或系统?等等。你想解决的问题,今天给大家介绍…

Spring基础学习——web

Spring基础学习——web 一、Spring整合Web环境1.1 JavaWeb三大组件作用及其特点1.2 Spring整合Web环境的思路及实现1.3 Spring开发Web环境组件spring-web1.4 web层MVC框架思想与设计思路 一、Spring整合Web环境 1.1 JavaWeb三大组件作用及其特点 在Java语言当中,w…

creo6.0教程之旋转,扫描

目录 一、旋转:二、扫描: 一、旋转: 案例1:旋转一个球: 任意一个平面绘制草图: 确定草图后,然后退出草图,点击旋转: 案例2:旋转一个杯子雏形: …

在以TAB为首地址的字存储区中存放有N个无符号数,试统计低3位全为1的数的个数(个数设为≤9),并显示。

;默认认采用ML6.11汇编程序 DATAS SEGMENT;此处输入数据段代码TAB DW -7,7,15,20,21N($-TAB)/2;G DW 0 DATAS ENDS STACKS SEGMENT;此处处输入堆栈段代码; DB 200 DUP(0) STACKS ENDS CODES SEGMENTASSUME CS:CODES,DS: DATAS, SS:STACKS START:MOV AX, DATASMOV DS,AX;此处输入…

swift和OC混编报错问题

1.‘objc’ instance method in extension of subclass of ‘xxx’ requires iOS 13.0.0 需要把实现从扩展移到主类实现。iOS13一下扩展不支持objc 2.using bridging headers with framework targets is unsupported 报错 这个错误通常指的是在一个框架目标中使用桥接头是不…

01:2440----点灯大师

目录 一:点亮一个LED 1:原理图 2:寄存器 3:2440的框架和启动过程 A:框架 B:启动过程 4:代码 5:ARM知识补充 6:c语言和汇编的应用 A:代码 B:分析汇编语言 C:内存空间 7:内部机制 二:点亮2个灯 三:流水灯 四:按键控制LED 1:原理图 2:寄存器配置 3:代码 一:点…

postgresql|数据库|提升查询性能的物化视图解析

前言: 我们一般认为数字的世界是一个虚拟的世界,OK,但我们其实有些需求是和现实世界一模一样的,比如,数据库尤其是关系型数据库,希望在使用的数据库能够更快(查询速度),…

亚马逊云AI应用科技创新下的Amazon SageMaker使用教程

目录 Amazon SageMaker简介 Amazon SageMaker在控制台的使用 模型的各项参数 pytorch训练绘图部分代码 Amazon SageMaker简介 亚马逊SageMaker是一种完全托管的机器学习服务。借助 SageMaker,数据科学家和开发人员可以快速、轻松地构建和训练机器学习模型&#…

765. 情侣牵手

765. 情侣牵手(leetcode,数学思维题)-------------------Java实现 题目表述 n 对情侣坐在连续排列的 2n 个座位上,想要牵到对方的手。 人和座位由一个整数数组 row 表示,其中 row[i] 是坐在第 i 个座位上的人的 ID。情侣们按顺…

头歌答案--爬虫实战

目录 urllib 爬虫 第1关:urllib基础 任务描述 第2关:urllib进阶 任务描述 requests 爬虫 第1关:requests 基础 任务描述 第2关:requests 进阶 任务描述 网页数据解析 第1关:XPath解析网页 任务描述 第…

汉明距离(Java)

两个整数之间的 汉明距离 指的是这两个数字对应二进制位不同的位置的数目。 给你两个整数 x 和 y,计算并返回它们之间的汉明距离。 方法1:使用内置函数 class Solution {public int hammingDistance(int x, int y) {return Integer.bitCount(x ^ y);} }方法2:移位实…

技能培训知识付费服务预约小程序的效果如何

技能、证书往往是很多人生活的基本,行业岗位竞争激烈,每个人都希望有多种技能或工作所需,而需求持续增加下,相关技能培训机构也很多,比如常见的考证、钢琴培训、针灸培训、花艺培训等。 很多行业都需要学习或考证&…

mac homebrew.mxcl.php@5.6.plist

今天启动php5.6时 遇到了一个问题 servers % brew services start php5.6 Bootstrap failed: 5: Input/output error Try re-running the command as root for richer errors. Error: Failure while executing; /bin/launchctl bootstrap gui/501 /Users/ssh/Library/LaunchAge…