《剑指offer》第24天:链表相加
01、题目示例
各数据结构中的“加减乘除”都是很爱考察的内容。
第2题:两数相加 |
---|
给出两个 非空 的链表用来表示两个非负的整数。其中,它们各自的位数是按照 逆序 的方式存储的,并且它们的每个节点只能存储 一位 数字。 |
给出两个 非空 的链表用来表示两个非负的整数。其中,它们各自的位数是按照 逆序 的方式存储的,并且它们的每个节点只能存储 一位 数字。
如果,我们将这两个数相加起来,则会返回一个新的链表来表示它们的和。
您可以假设除了数字 0 之外,这两个数都不会以 0 开头。
示例:
输入:(2 -> 4 -> 3) + (5 -> 6 -> 4)
输出:7 -> 0 -> 8
原因:342 + 465 = 807
02、算法图解
烧饼(哨兵)节点:烧饼节点就是说有一个节点长的像烧饼,站在最前面。往往第一个节点小子都会来啃一口,因为是牛肉拔丝烧饼,扯不掉就连起来了。。。好了,我实在编不下去了~囧....
不了解哨兵节点的可以看下:
好了,了解了哨兵节点,但是这道题和哨兵节点并没有什么毛线关系,我只是让你复习一下(会不会想打我)。现在开始正式分析题目。
老样子,我们还是先画图,假若我们的链表是 (2 -> 4 -> 3) + (5 -> 6 -> 4) ,大概就是这样(注意,这里我们其实是要完成 342 + 465):
加法肯定是从最低位到最高位进行相加,也就是这里的链表头到链表尾进行相加,所以需要遍历链表。我们令 l1 和 l2 指向两个链表的头,用一个 tmp 值来存储同一位相加的结果,以及一个新的链表来存储 tmp 的值。(为什么不直接用新链表存储结果,而是用 tmp 先存储结果,再用新链表存储 tmp 的值呢?请继续往下看)
记住:所有模拟运算的题目,都需要考虑进位。这里因为个位不涉及进位,开始计算十位。我们同时遍历 l1,l2,以及新链表到下一位。
重复上面的操作,可以看到十位与个位不同,发生了进位。这时,刚才的 tmp 就有了用武之地。我们使用 tmp 携带进位的值到下一位的运算。自然,这里的链表也不能直接存储 tmp 的值了,而是要存储 tmp%10 的值。重复这个步骤,直到两个链表都遍历完成,并且 tmp 没有进位值:
现在只需要返回我们的新链表就可以了。问题来了,因为我们没有构造哨兵节点,所以此时不太容易直接返回新链表。所以在整个流程的第一步,我们还需要用一个哨兵节点指向我们的新链表。
分析完毕,直接上代码,go:
//go
func addTwoNumbers(l1 *ListNode, l2 *ListNode) *ListNode {
list := &ListNode{0, nil}
//这里用一个result,只是为了后面返回节点方便,并无他用
result := list
tmp := 0
for l1 != nil || l2 != nil || tmp != 0 {
if l1 != nil {
tmp += l1.Val
l1 = l1.Next
}
if l2 != nil {
tmp += l2.Val
l2 = l2.Next
}
list.Next = &ListNode{tmp % 10, nil}
tmp = tmp / 10
list = list.Next
}
return result.Next
}
执行结果:
java:
class Solution {
public ListNode addTwoNumbers(ListNode l1, ListNode l2) {
ListNode listNode = new ListNode(0);
ListNode result = listNode;
int tmp = 0;
while (l1 != null || l2 != null || tmp != 0) {
if (l1 != null) {
tmp += l1.val;
l1 = l1.next;
}
if (l2 != null) {
tmp += l2.val;
l2 = l2.next;
}
listNode.next = new ListNode(tmp % 10);
tmp = tmp / 10;
listNode = listNode.next;
}
return result.next;
}
}
执行结果:
- 一个人的武林:内网渗透测试思路(二)
- Python数据分析之股票实战
- 这货不是电源:硬件渗透测试平台 – Power Pwn
- Python进阶学习之阅读代码
- 代码安全审计:当file_exists遇上eval
- Python 异常处理完整指南
- 有趣的安全实验:利用多线程资源竞争技术上传shell
- NotSerializableException解决方法NotSerializableException
- 谁蹭了我的WiFi?浅谈家用无线路由器攻防
- 当输入流和输出流同时作用一个文件
- 揭秘:充电宝是如何盗取你的个人隐私的?
- 微软:暴力破解面前,增强密码复杂性基本没用
- 创建被图像填充的组件解释几处做法解释几点
- Android原生嵌入React Native
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- Oracle基本参数(DB_CREATE_FILE_DEST,DB_CREATE_ONLINE_LOG_DEST_n)
- React16的memo函数有啥用
- 要点2:循环、条件控制
- VSCode代码格式化设置
- MapReduce之MapJoin案例
- MapReduce之多个Job串联的案例
- Java Class文件常量池
- Kubernetes 学习笔记——使用 Heml 安装和使用 OpenFaaS
- 奈学:reaseShared共享式释放锁
- 奈学:reaseShared共享式释放锁
- 【DB宝18】在Docker中安装使用MySQL高可用之MGR
- MapStruct
- 聊聊claudb的exportRDB
- Jenkins_流水线语法_001
- Jenkins_流水线语法_002