链表算法题之中等级别,debug调试更简单
文章简述
大家好,本篇是个人的第 5 篇文章
从本篇文章开始,分享关于链表的题目为中等难度,本次共有 3 道题目。

一,两数相加

1.1 题目分析
题中写到数字是按照逆序的方式存储,从进位的角度看,两两节点相加我们是可以直接将进位传递到下一组两两节点相加。
比如题中第二组节点【4】和节点【6】相加结果为 10,而 10 就需要进位,也就是说该节点只能保存数字【0】,而进位【1】就要传递到下一组节点相加。
那再整理下思路。
如果两个链表的节点数是相等的,那只需要依次将两两节点进行相加。如果节点数是不相等的,比如。

L2 链表少了一个节点,像这种情况我们就需要在高位用【0】进行补位。

我们再回到题中的案例,而上面说的位数不够也是需要考虑的一种情况。再一步步分析下如何进行两两节点相加。

第一组节点相加为2+5=7,不满足进位。创建一个新的链表保存相加后的数,那此时链表第一个节点数为【7】。
接着是4+6=10,此时满足进位要求,按照题目要求和我们上面的分析,需要将低位【0】保存到节点,高位【1】传递到下一组节点。

那现在进行最后一组相加3+4=7,但是还有重要一步不能丢,即上一组节点相加时,还有高位进 1,那最后的结果是 3+4+1=8。
最后将上面相加的结果用链表进行保存,那么结果为。

同理,如果位数不足时用【0】进行补位也是一样的方式。
1.2 代码分析
老方式,先创建单链表
// 创建链表-L1
ListNode l1 = new ListNode(2);
ListNode l2 = new ListNode(4);
ListNode l3 = new ListNode(3);
ListNodeFun listNodeFun = new ListNodeFun();
listNodeFun.add(l1);
listNodeFun.add(l2);
ListNode listNode1 = listNodeFun.add(l3);
ListNodeFun listNodeFun2 = new ListNodeFun();
// 创建链表-L2
ListNode l11 = new ListNode(5);
ListNode l22 = new ListNode(6);
ListNode l33 = new ListNode(4);
listNodeFun2.add(l11);
listNodeFun2.add(l22);
ListNode listNode2 = listNodeFun2.add(l33);
两数相加代码
if(null == l1 || null == l2){
return null;
}
// 初始化头指针
ListNode head = new ListNode();
ListNode cur = head;
// 定义变量保存进位值
int temp = 0;
while(null != l1 || null != l2){
// 获取每个节点的值
int x = l1 == null ? 0 : l1.val;
int y = l2 == null ? 0 : l2.val;
// 两数相加
int sum = x + y + temp;
// 获取相加结果
temp = sum / 10;
// 获取低位(个位)
sum = sum % 10;
// 创建新的节点
cur.next = new ListNode(sum);
// 移动指针
cur = cur.next;
// 移动链表指针,要判断为空,否则会空针
if(null != l1){
l1 = l1.next;
}
if(null != l2){
l2 = l2.next;
}
}
if(1 == temp){
cur.next = new ListNode(1);
}
return head.next;
}
1.3 debug 调试
第一步,2+5
两个链表的首节点相加,结果为 7。
【7】不需要进位,创建新的链表进行保存。

第二步,4+6

结果为 10 就需要进位,将个位上的【0】保存到节点中,十位上【1】需要进行进位。
第三步,3+4+1

最后看下运行结果

简单总结下,这道题并不算难,但需要考虑清楚当节点相加时是否需要进行补位的情况。
二,删除链表的倒数第 N 个结点

2.1 题目分析
这道题,是不是似曾相识?

没错,在上一篇文章中《链表算法题二,还原题目,用 debug 调试搞懂每一道题》有一道题是【链表中倒数第 k 个节点】。但是这两道题之间略有不同,上一篇文章中的题目是返回倒数第 K 个节点,本道题中是移除第 K 个节点,返回其他完整链表。
那么这两道题相似度很高,是不是套路也是一样。
上一道题我们使用了双指针的方式,那本道题也是一样的。所以上一道题如果搞懂了,那这道所谓中等级别的题也就成简单级别的了。虽然本人目前题量不多,但是如果善于总结的话,套路确实很接近,反正这个题我是直接写出来了,哈哈(开玩笑)。

话又说回来,分析题中的含义,假设移除节点【4】,按照双指针的方式,那就是一个慢指针指向节点【3】,快指针指向节点【5】。将节点【3】的下下个 next 指向节点【5】,即可移除节点【4】。

参考上一道题的方式,需要将【fast】快指针先移动 K 个节点,初始化指针位置。
注意:移除节点后,是需要反回其它完整的链表节点。但是有一种情况有坑,先看下图

链表只有一个节点并移除,正确结果应该是返回空。像这种情况是不能直接返回 head 链表,因此是需要创建头指针来指引原始的链表,如下图。

所以定义双指针的起始节点位置就是 head 节点。
按照套路先将快指针移动 K 个节点

剩下的操作即移动快慢指针,直到 fast 指针移动到最后一个节点。
(1)

(2)

(3)

最后更改 slow 指针直接指向 fast 指针指向的节点即可。
2.2 代码分析
创建链表的代码同上题一样,本道题只需要创建一个 1-5 节点的链表。
直接贴上删除节点的代码
public ListNode removeNthFromEnd(ListNode head, int n) {
ListNode pre = new ListNode();
pre.next = head;
// 定义双指针
ListNode slow = pre;
ListNode fast = head;
// 先将快指针移动n个节点
while(--n>0){
fast = fast.next;
}
while(fast.next != null){
fast = fast.next;
slow = slow.next;
}
slow.next = slow.next.next;
return pre.next;
}
2.3 debug 调试
我们先 debug 调试看下初始化节点位置后,快慢指针的位置。

接着进入第 2 个 while 循环,将剩余的节点遍历完。

这一步 slow 指针到节点【1】,fast 指针到节点【3】
下一步 slow 指针到节点【2】,fast 指针到节点【4】,直接看最后一步 slow 指针到节点【3】,fast 指针到节点【5】

节点【5】即最后一个节点,此时退出循环,最后将 slow 指针 next 指向 fast 指针指向的节点。

运行结果

三,两两交互链表中的节点

3.1 题目分析
说来惭愧,这道题当时写的时候基本没有什么思路,结果还是看了题解才写出来的。

现在想想这真是道灵魂题啊,真是没想到还能用递归去写这道题,不得不说真是万能的递归啊(主要本人太菜,哈哈)。
递归的方式在于如果是偶数链表,将两两节点相互交换;如果是奇数链表,那最后一个节点保持不动,下面用 debug 调试会看的清楚些。
将在偶数位上的节点指向上一个奇数位的节点,使用递归依次类推来遍历整个链表。
大致的思路是这样,使用递归将链表遍历结束,然后返回最后节点【4】并指向上一个节点【3】;接着返回递归的结果【2】指向上一个节点【1】,而节点【1】也是指向节点【4】。

接着下次递归

按照规则分析,节点【4】与节点【3】交换位置,那返回的就是节点【4】

现在回到第一步递归的结果,当时 head 指向的节点【1】,那么 head.next 指向谁?现在递归结果返回节点【4】,因此 head.next 也就指向的是节点【4】
最后节点【1】与节点【2】交换位置,就成了最后的链表交换结果。
这样分析还是很抽象,下面用 debug 调试走一遍就清晰了。
3.2 代码分析
递归的代码还是比较简单,先贴上来。
public ListNode swapPairs(ListNode head) {
if(head == null || head.next == null){
return head;
}
ListNode nextNode = head.next;
head.next = swapPairs(nextNode.next);
nextNode.next = head;
return nextNode;
}
3.3 debug 调试
第一步,节点【1】和节点【2】

开始进入递归循环,此时 nextNode 节点为【2】,那该节点的下一个节点为【3】
第二步,节点【3】和节点【4】

现在 nextNode 节点为【4】,再次进入递归循环时,节点【4】的 next 就为 null,因为节点【4】为最后一个节点,开始结束递归。

现在开始返回递归的结果,首先返回的就是节点【3】和节点【4】

再看第 43 行代码,将节点【4】下一个节点指向了节点【3】,并返回了节点【4】

接着返回节点【1】和节点【2】

注意:上一步递归中,我们返回的结果为节点【4】
上图中看到 head 节点为【1】,而 head.next 也就是节点【4】了

最后返回交换后的节点【1】

3.4 小补充
还记得上面我们说的,如果链表为奇数,最后结果如何呢?
现在接着上面的 debug 看下最后奇数的节点怎么返回。
假设现在新增一个节点【5】

按照 if 判断,节点【5】为最后一个节点,进入 if 判断后就将节点【5】返回。
还记得我们上面说的 head.next 指针吗,它指向的是递归返回的结果,我们最后一次递归的时候,head 不就是节点【3】吗!


当节点【3】和节点【4】交换后,节点【3】不就正好指向了返回的节点【5】

四,总结
解决链表相关的题目,我们大多可以使用双指针(快慢指针),数组,递归,迭代这 4 种方式。
在做完简单题目后,再加上本篇文章的 3 道中等题目,使用双指针,递归就可解决大多数的题目。后面将中等题目刷完后,再来看看链表题目有多少是可以用上述几种方式去解决。
最后,求关注
原创不易,每一篇都是用心在写。如果对您有帮助,就请一键三连(关注,点赞,再转发)
我是杨小鑫,坚持写作,分享更多有意义的文章。
感谢您的阅读,期待与您相识!

链表算法题之中等级别,debug调试更简单的更多相关文章
- 灵活使用 console 让 js 调试更简单
摘要: 玩转console. 原文:灵活使用 console 让 js 调试更简单 作者:前端小智 Fundebug经授权转载,版权归原作者所有. Web 开发最常用的就是 console.log , ...
- vue—你必须知道的 js数据类型 前端学习 CSS 居中 事件委托和this 让js调试更简单—console AMD && CMD 模式识别课程笔记(一) web攻击 web安全之XSS JSONP && CORS css 定位 react小结
vue—你必须知道的 目录 更多总结 猛戳这里 属性与方法 语法 计算属性 特殊属性 vue 样式绑定 vue事件处理器 表单控件绑定 父子组件通信 过渡效果 vue经验总结 javascript ...
- [转]九个Console命令,让js调试更简单
转自:九个Console命令,让js调试更简单 一.显示信息的命令 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 <!DOCTYPE html> <html ...
- 分享几个日常调试方法让js调试更简单
下面分享几个日常调试代码的时候在Console命令行显示你的操作,让你的js调试更简单. console显示信息的命令 在浏览器按f12在console上显示你的文本. <!DOCTYPE ht ...
- 让js调试更简单—console
一.显示信息的命令 console.log 用于输出普通信息 console.info 用于输出提示性信息 console.error用于输出错误信息 console.warn用于输出警示信息 最常用 ...
- 【F12】Console命令,让js调试更简单
Console命令,让js调试更简单 一.显示信息的命令 console.log("normal"); // 用于输出普通信息 console.info("informa ...
- .net core2.0添加json文件并转化成类注入控制器使用 让js调试更简单—console
.net core2.0添加json文件并转化成类注入控制器使用 上一篇,我们介绍了如何读取自定义的json文件,数据是读取出来了,只是处理的时候太麻烦,需要一遍一遍写,很枯燥.那么有没有很好的办法呢 ...
- 链表算法题二,还原题目,用debug调试搞懂每一道题
文章简述 大家好,本篇是个人的第4篇文章. 承接第3篇文章<开启算法之路,还原题目,用debug调试搞懂每一道题>,本篇文章继续分享关于链表的算法题目. 本篇文章共有5道题目 一,反转链表 ...
- LeetCode 上最难的链表算法题,没有之一!
题目来源于 LeetCode 第 23 号问题:合并 K 个排序链表. 该题在 LeetCode 官网上有关于链表的问题中标注为最难的一道题目:难度为 Hard ,通过率在链表 Hard 级别目前最低 ...
随机推荐
- Codeforces Round #643 (Div. 2) E. Restorer Distance (贪心,三分)
题意:给你\(n\)个数,每次可以使某个数++,--,或使某个数--另一个++,分别消耗\(a,r,m\).求使所有数相同最少的消耗. 题解:因为答案不是单调的,所以不能二分,但不难发现,答案只有一个 ...
- Musical Theme POJ - 1743 后缀数组
A musical melody is represented as a sequence of N (1<=N<=20000)notes that are integers in the ...
- 吉哥系列故事――恨7不成妻 HDU - 4507
题目: 单身! 依然单身! 吉哥依然单身! DS级码农吉哥依然单身! 所以,他生平最恨情人节,不管是214还是77,他都讨厌! 吉哥观察了214和77这两个数,发现: 2+1+4=7 7+7=7*2 ...
- CF1397-C. Multiples of Length
CF1397-C. Multiples of Length 题意: 给出一个长度为\(n\)的序列,让你进行下面操作三次使得整个序列全部变为\(0\): 在序列中选中一段序列\((l, r)\), ...
- 国产网络损伤仪 SandStorm -- 只需要拖拽就能删除链路规则
国产网络损伤仪SandStorm可以模拟出带宽限制.时延.时延抖动.丢包.乱序.重复报文.误码.拥塞等网络状况,在实验室条件下准确可靠地测试出网络应用在真实网络环境中的性能,以帮助应用程序在上线部署前 ...
- XV6学习(15)Lab mmap: Mmap
代码在Github上. 这一个实验是要实现最基础的mmap功能.mmap即内存映射文件,将一个文件直接映射到内存当中,之后对文件的读写就可以直接通过对内存进行读写来进行,而对文件的同步则由操作系统来负 ...
- Linux-开机运行流程
目录 CentOS7开机流程 Linux运行级别 systemd进程管理 systemd的优势 systemd相关文件 systemd启动相关命令 systemd开机自启动相关命令 systemd服务 ...
- Linux-字符处理命令
目录 1.sort(排序) 2.uniq(不相邻的两行重复不会去除) 3.cut(取列,截取字段) 4.wc(统计行.单词.字符数) 1.sort(排序) 选项: -t # 指定分隔符 -k # 指定 ...
- mysql+python+pymysql的一些细节问题
报错 (1044, "Access denied for user 'erio'@'localhost' to database 'library'") 就是权限问题了,没什么好说 ...
- 流水线cpu —Verilog HDL
一.准备工作 先看看书(<计算机原理与设计 Verilog HDL版>),搞懂一点原理.然后照着书上的代码写一写(用8.4的就可以了,不用8.6的). 注意mux2x32,mux4,cla ...