当前位置: 首页 > news >正文

回文链表(快慢指针解法之在推进过程中反转)

归纳编程学习的感悟,
记录奋斗路上的点滴,
希望能帮到一样刻苦的你!
如有不足欢迎指正!
共同学习交流!
🌎欢迎各位→点赞 👍+ 收藏⭐ + 留言​📝

抱怨深处黑暗,不如提灯前行!

 

题目描述:

给你一个单链表的头节点 head ,请你判断该链表是否为回文链表。如果是,返回 true ;否则,返回 false 。

示例 1:

输入:head = [1,2,2,1]
输出:true

示例 2:

输入:head = [1,2]
输出:false

提示:

  • 链表中节点数目在范围[1, 105] 内
  • 0 <= Node.val <= 9

思路推导

判断该链表是否为回文链表,数据范围node.val<=9,节点数量∈[1,10^5]node.val <= 9, 节点数量 ∈ [1,10^5]node.val<=9,节点数量∈[1,10^5 ]

可选方案1:转数字,判断是否为回文数
可选方案2:转字符串,判断是否为回文串
但这样做就没有利用到链表自身的性质,因此不是最合适的解法。
根据「206.反转链表」的思路,可以在完成「链表前半部分的反转」之后,通过双指针指向前半部分的头结点和后半部分链表的头结点,每次推进一步并判断数值是否相等。
但这存在「奇数节点个数的链表」和「偶数节点个数的链表」的前半部分和后半部分划分上的不同的问题。
情况分析
回文链表必然会存在「奇数个节点」的链表和「偶数个节点」的链表。

对于长度为奇数的回文链表
1->2->3->2->1,长度为5的链表,那么从回文中心3向左右两侧出发,会遍历得到相同的数字序列。
对于偶数长度的回文链表:
1->2->2->1,长度为4的链表,如果回文,那么从第2和第3个节点出发,向两侧扫描,可以得到相同的数字序列。
因此,假设节点个数为n

奇数链表:

回文中心为:(n+1)/2个节点
回文边界为:(n+1)/2 - 1,(n+1)/2 + 1
偶数链表:

没有回文中心
回文边界:n/2和n/2 + 1
之后可以通过「统计节点个数」->「对奇偶链表需要翻转的节点个数分别判断完成前半部分反转」->「从前半部分链表和后半部分链表的头结点向两侧扫描推进判断数字是否相同」的方法来判断回文链表。

但目前这样的考虑过于复杂了,因为实际上我们无需计算节点个数,只需要确定链表中间节点的位置即可。「确定链表中间节点的位置」的方法通常使用快慢指针法。

示例分析
设计双指针slow = head, fast = head
「奇数链表」和「偶数链表」最终确定的中间节点不一致,我们可以先确定「链表中间节点位置」,再「反转前半部分的链表」;也可以考虑在slow指针推进过程中完成「局部反转」,这样到达「中间位置节点」时恰好完成了前半部分的反转。

  • 偶数链表的示例分析:
局部翻转还需要的变量pre以及变量初始化:
设计pre = null, slow = fast = head;
当前循环条件未知。
每次循环,slow指针每次前进一步,就翻转局部链表,fast指针前进两步。
偶数节点情况:
初始化:
null 1 -> 2 -> 2 -> 1 -> null^   ^
pre slow,fast
---------------------------------
第1轮循环
null<- 1   2 ->  2 -> 1 -> null^   ^ 	 ^pre  slow fast
---------------------------------
第2轮循环:
null<- 1 <- 2   2-> 1-> null^	^		 ^pre  slow 	fast
第二次循环结束后,就已经完成了前半部分链表的反转,
pre指向前半部分链表的第一个元素。
slow到达「中间节点的第二个节点」。
slow指针指向后半部分链表的第一个元素。
偶数链表的循环退出条件是 fast == null
  • 奇数链表的示例分析:

奇数链表模拟:
初始化
null 1 -> 2 -> 3 -> 2 -> 1 -> null^   ^
pre  slow,fast
第一轮循环
null<- 1	2 -> 3 -> 2 -> 1 -> null^    ^	 ^pre  slow fast第二轮循环:
null<- 1 <- 2 	3 -> 2 -> 1 -> null^	^		  ^pre slow		fast
第二轮循环结束后,slow到达「链表中间节点」位置。
循环退出条件是 fast.next == null 
循环退出时
fast = 链表最后一个元素
pre 指向前部分链表的第一个元素
slow 指针指向后部分链表的第一个元素。
需要特殊处理:即将slow指针向后推进一位,
以保证和「偶数链表」相同,slow指针指向后半部分回文链表的第一个元素。
  • 抽取循环结束时的共性:

pre指向前半部分链表的第一个元素。
slow指针经过处理后,指向后半部分链表的首个元素
循环退出条件:
        对于奇数链表为fast.next == null
        对于偶数链表为fast==null

  • 通过「奇数链表」和「偶数链表」的示例分析得到伪代码:

循环开始:
1. 变量初始化 
双指针赋值slow = head, fast = head, 
局部旋转前一个变量指针pre = null
局部翻转需要保存的下一个节点 tmp = null 
2. 双指针推进循环
循环条件(fast != null && fast.next != null){2.1 快指针每次推进两步。2.2 慢指针通过「局部反转」的方式进行指针的推进。
}
3.循环结束后的情况分析:3.1 偶数链表循环结束后,fast == null,pre指向前半部分第一个回文数slow指向后半部分第一个回文数3.2 奇数链表循环结束后,fast !=null, fast.next == nullpre指向前半部分第一个回文数slow指向后半部分第一个节点(链表中间节点)。为使得一致,slow指针需要指向第一个回文数需要将slow向前推进一步
因此,如果fast != null,将slow指针推进一步。
4. 此时pre指向前半部分链表第一个回文数,slow指向后半部分链表第一个回文数。
循环判断:
循环条件:pre!= null || slow.next != null
向两边同时推进pre和slow,比较pre.val和slow.val是否相等,
如果pre.val != slow.val,说明不是回文数,返回false。
当pre和slow都推进到链表末尾,说明每个数都相等,返回true。

代码实现:

/*** Definition for singly-linked list.* struct ListNode {*     int val;*     struct ListNode *next;* };*/
typedef struct ListNode LNode;
bool isPalindrome(struct ListNode* head) {LNode*fast;LNode*slow;fast=head;slow=head;while(fast&&fast->next){fast=fast->next->next;slow=slow->next;}LNode *temp,*pre,*cur;pre=NULL;cur=slow;while(cur){temp=cur->next;cur->next=pre;pre=cur;cur=temp;}LNode*p;p=head;while(pre&&p){if(p->val!=pre->val){break;}p=p->next;pre=pre->next;}if(p==NULL||pre==NULL){return true;}else{return false;}
}

复杂度分析:

时间复杂度:O(n),双指针扫描链表一遍O(n),局部反转时间复杂度O(1),回文数比较时间复杂度O(n)。
空间复杂度:O(1),指针只占用了常量的空间。

相关文章:

  • 《当微服务遇上Ribbon:一场负载均衡的华丽舞会》
  • Android Studio 所有历史版本下载
  • P1160 队列安排
  • Excel插入多行VBA实现
  • 【Real】[Flask]SSTI
  • 2024年,企业人才招聘怎么做?
  • B站pink老师HTML5基础(一)
  • SAP OBYC自动记账 详解
  • LangChain笔记
  • makefile一些特殊且常用的符号
  • 哈希算法教程(个人总结版)
  • 查询DQL
  • 赶紧收藏!2024 年最常见 20道 Rocket MQ面试题(二)
  • 基于python flask +pyecharts实现的气象数据可视化分析大屏
  • NDIS小端口驱动开发(一)
  • 分享一款快速APP功能测试工具
  • CSS实用技巧
  • iBatis和MyBatis在使用ResultMap对应关系时的区别
  • SpiderData 2019年2月16日 DApp数据排行榜
  • SwizzleMethod 黑魔法
  • 世界编程语言排行榜2008年06月(ActionScript 挺进20强)
  • 在Docker Swarm上部署Apache Storm:第1部分
  • 中国人寿如何基于容器搭建金融PaaS云平台
  • zabbix3.2监控linux磁盘IO
  • 阿里云API、SDK和CLI应用实践方案
  • 仓管云——企业云erp功能有哪些?
  • # centos7下FFmpeg环境部署记录
  • # SpringBoot 如何让指定的Bean先加载
  • #define MODIFY_REG(REG, CLEARMASK, SETMASK)
  • (1)(1.19) TeraRanger One/EVO测距仪
  • (1)Jupyter Notebook 下载及安装
  • (6)设计一个TimeMap
  • (Demo分享)利用原生JavaScript-随机数-实现做一个烟花案例
  • (done) 两个矩阵 “相似” 是什么意思?
  • (vue)页面文件上传获取:action地址
  • (免费领源码)python#django#mysql公交线路查询系统85021- 计算机毕业设计项目选题推荐
  • (七)glDrawArry绘制
  • (使用vite搭建vue3项目(vite + vue3 + vue router + pinia + element plus))
  • (四)stm32之通信协议
  • (一)【Jmeter】JDK及Jmeter的安装部署及简单配置
  • (已解决)什么是vue导航守卫
  • **Java有哪些悲观锁的实现_乐观锁、悲观锁、Redis分布式锁和Zookeeper分布式锁的实现以及流程原理...
  • .form文件_SSM框架文件上传篇
  • .mp4格式的视频为何不能通过video标签在chrome浏览器中播放?
  • .NET Entity FrameWork 总结 ,在项目中用处个人感觉不大。适合初级用用,不涉及到与数据库通信。
  • .net wcf memory gates checking failed
  • .net 生成二级域名
  • .net 使用ajax控件后如何调用前端脚本
  • ::什么意思
  • @Service注解让spring找到你的Service bean
  • @软考考生,这份软考高分攻略你须知道
  • [.net] 如何在mail的加入正文显示图片
  • [\u4e00-\u9fa5] //匹配中文字符
  • []Telit UC864E 拨号上网
  • [001-03-007].第07节:Redis中的管道