Leetcode 002. 两数相加
1.题目描述
给出两个 非空 的链表用来表示两个非负的整数。其中,它们各自的位数是按照 逆序 的方式存储的,并且它们的每个节点只能存储 一位 数字。
如果,我们将这两个数相加起来,则会返回一个新的链表来表示它们的和。
您可以假设除了数字 0 之外,这两个数都不会以 0 开头。
示例:
输入:(2 -> 4 -> 3) + (5 -> 6 -> 4)
输出:7 -> 0 -> 8
原因:342 + 465 = 807
2.我的错误版本
2.1 解题思路
题中链表顺序正好是低位到高位,先低位相加,有进位保持,下一位计算加上进位,直到最高位相加,如果有进位,生成新的结点保存。
2.2 我的错误代码
有时候,从错误中才能学到真知识。
现在刷题,很容易到网上找到各种答案,因此,初学者往往因为无法忍受一遍又一遍的调试,而放弃独立思考,选择取拷贝别人的代码,以理解别人的代码代替了自己刷题的那个拔高的训练过程,这实在是很可惜的,刷题的效果会大打折扣。
我的错误:
- 应当创建新的链表来保存结果,保留原链表。因为传入原链表L1和L2可能还需要用到,不能轻易更改;
- 解题思路不对:
- a.用了繁琐的思路
- b.用了繁琐的思路但没有列举全所有可能的情况:(1)两个链表一样长;(2)L1比L2长;(3)L2比L1长;(4)两个链表均为单结点,如5和5,这个测试用例反复修改,过不了,最后发现要换思路才能过。(方法不对,努力白费)
- c.链表遍历,不用遍历到NULL,处理NULL很繁琐。
//执行结果错误,方法不对,努力白费,用这个思路改了好多遍,放弃了这个错误或繁琐的思路。
//没有创建全新的链表,使用L1、L2中较长的保存结果
class Solution {
public:
ListNode* addTwoNumbers(ListNode* l1, ListNode* l2) { unsigned int count1 = , count2 = ;//统计两个链表长度
unsigned int carry = ; //进位 while (l1 != NULL && l2 != NULL)
{
l1->val = (l1->val + l2->val + carry) % ;
l2->val = (l1->val + l2->val + carry) % ; //判断有无进位
if ((l1->val + l2->val + carry) / != )
carry = ;
else
carry = ; // ((l1->val + l2->val + carry) / 10 != 0)?carry = 1 : carry = 0 ; l1 = l1->next;
l2 = l2->next;
} if (l1 != NULL)
{ //L1更长,剩余链表处理
while (l1 != NULL)
{
l1->val = (l1->val + carry) % ;
if ((l1->val + carry) / != )
carry = ;
else
carry = ;
l1 = l1->next;
}
if (l1 == NULL && carry == )
ListNode();
return l1;
}
else
{ //L2更长,剩余链表处理
while (l2 != NULL)
{
l2->val = (l2->val + carry) % ;
if ((l2->val + carry) / != )
carry = ;
else
carry = ;
l2 = l2->next;
}
if (l2 == NULL && carry == )
ListNode();
return l2;
} }
};
3.正确简洁版本
好的代码应该用比较通用的形式,概括表达出所有可能的情况。代码中红色标注的行是关键行。
//一般地,建议建立新链表保存结果,不在原链表上直接做修改,因为原链表可能还要使用。 #include <iostream>
#include <vector> using namespace std; struct ListNode {
int val;
ListNode *next;
ListNode(int x) : val(x), next(NULL) {}
}; typedef ListNode* LinkList; class Solution {
public:
ListNode* addTwoNumbers(ListNode* l1, ListNode* l2) { unsigned int carry = ; //进位 ListNode* Hl3 = new ListNode();//头节点
ListNode* l3 = Hl3; int a = ;
int b = ;
int temp = ; while (l1 != NULL || l2 != NULL)
{
//判断a和b的取值
if (l1 == NULL) a = ;
else a = l1->val;
if (l2 == NULL) b = ;
else b = l2->val; //a = (l1==NULL?) 0 : l1->val;
//b = (l2==NULL?) 0 : l2->val;
temp = (a + b + carry) % 10; //有无进位
if ((a + b + carry) / != )
carry = ;
else
carry = ; //创建新结点,保存temp
l3->next = new ListNode(temp);
l3 = l3->next; if (l1) l1 = l1->next;
if (l2) l2 = l2->next;
} //最后的进位
if (carry == ) {
l3->next = new ListNode();
} return Hl3->next; }
}; int main()
{
ListNode* p1 = new ListNode();
ListNode* p2 = new ListNode();
ListNode* p3 = new ListNode();
p1->next = p2;
p2->next = p3;
p3->next = NULL; ListNode* q1 = new ListNode();
ListNode* q2 = new ListNode();
ListNode* q3 = new ListNode();
q1->next = q2;
q2->next = q3;
q3->next = NULL; Solution test;
ListNode* t1 = test.addTwoNumbers(p1, q1);
while (t1) {
cout << t1->val << endl;
t1 = t1->next;
}
return ;
} //可能的问题:内存泄漏,创建的头节点Hl3最终没有被释放。
4.用时更少的范例
//来源:Leetcode官网提交的答案
//基本思路和方式是相似的 /*
*
* Definition for singly-linked list.
* struct ListNode {
* int val;
* ListNode *next;
* ListNode(int x) : val(x), next(NULL) {}
* };
*/ //优化C++I/O提速
static const auto __ = []() {
ios::sync_with_stdio(false);
cin.tie(nullptr);
return nullptr;
}(); class Solution {
public:
ListNode* addTwoNumbers(ListNode* l1, ListNode* l2) {
ListNode* L_result = new ListNode();
ListNode* a = l1;
ListNode* b = l2;
ListNode*cur = L_result;
int carry = ;
while(a != NULL || b != NULL)
{
int val1 = a != NULL ? a->val : ;
int val2 = b != NULL ? b->val : ;
int sum = val1 + val2 + carry;
carry = bool(sum / );
cur->next = new ListNode(sum %); //先判是否为空,再赋值
if(a != NULL)
{
a = a->next;
}
if(b != NULL)
{
b = b->next;
}
cur = cur->next;
}
if(carry)
{
cur->next = new ListNode();
}
return L_result->next; }
}; //可能的问题:内存泄漏,创建的头节点L_result最终没有被释放。
5.两数相加 II
跳转另一篇博客:Leetcode 445. 两数相加 II
参考资料:
1. [LeetCode] Add Two Numbers 两个数字相加
Leetcode 002. 两数相加的更多相关文章
- LeetCode 445——两数相加 II
1. 题目 2. 解答 2.1 方法一 在 LeetCode 206--反转链表 和 LeetCode 2--两数相加 的基础上,先对两个链表进行反转,然后求出和后再进行反转即可. /** * Def ...
- Leetcode 445. 两数相加 II
1.题目描述 给定两个非空链表来代表两个非负整数.数字最高位位于链表开始位置.它们的每个节点只存储单个数字.将这两数相加会返回一个新的链表. 你可以假设除了数字 0 之外,这两个数字都不会以零开头. ...
- LeetCode 445. 两数相加 II(Add Two Numbers II)
445. 两数相加 II 445. Add Two Numbers II 题目描述 给定两个非空链表来代表两个非负整数.数字最高位位于链表开始位置.它们的每个节点只存储单个数字.将这两数相加会返回一个 ...
- LeetCode 2. 两数相加(Add Two Numbers)
2. 两数相加 2. Add Two Numbers 题目描述 You are given two non-empty linked lists representing two non-negati ...
- LeetCode 2——两数相加(JAVA)
给出两个 非空 的链表用来表示两个非负的整数.其中,它们各自的位数是按照 逆序 的方式存储的,并且它们的每个节点只能存储 一位 数字. 如果,我们将这两个数相加起来,则会返回一个新的链表来表示它们的和 ...
- Java实现 LeetCode 2 两数相加
两数相加 给出两个 非空 的链表用来表示两个非负的整数.其中,它们各自的位数是按照 逆序 的方式存储的,并且它们的每个节点只能存储 一位 数字. 如果,我们将这两个数相加起来,则会返回一个新的链表来表 ...
- leetcode TOP100 两数相加
两数相加 给出两个 非空 的链表用来表示两个非负的整数.其中,它们各自的位数是按照 逆序 的方式存储的,并且它们的每个节点只能存储 一位 数字. 如果,我们将这两个数相加起来,则会返回一个新的链表来表 ...
- 【LeetCode】两数相加
题目描述 给出两个非空的链表用来表示两个非负的整数.其中,它们各自的位数是按照逆序的方式存储的,并且它们的每个节点只能存储一位数字. 如果,我们将这两个数相加起来,则会返回一个新的链表来表示它们的和. ...
- leetcode 链表 两数相加
两数相加 给定两个非空链表来表示两个非负整数.位数按照逆序方式存储,它们的每个节点只存储单个数字.将两数相加返回一个新的链表. 你可以假设除了数字 0 之外,这两个数字都不会以零开头. 示例 ...
随机推荐
- lr 常用操作
lr脚本编写语法: web_add_cookie();:服务器注入cookies lr_save_string("网址或其他","参数2");:一个保存函数,它 ...
- 【第一章】MySQL数据概述
安装部署 备份恢复主备复制读写分离HA架构分布式数据库压力测试性能优化自动化运维 ==数据的存储方式1. 人工管理阶段2. 文件系统阶段3. 数据库系统管理阶段 ==数据库技术构成1. 数据库系统 D ...
- 【MySQL解惑笔记】Navicat 无法远程连接MySQL数据库
安装好Navicat之后远程连接MySQL数据库出现以下报错截图: 出现以上截图怀疑是mysql用户权限不够: GRANT ALL PRIVILEGES ON *.* TO 'root'@'192.1 ...
- ServiceStack.Ormlit 使用Insert的时候自增列不会被赋值
Insert签名是这样的,将第2个参数设置为true就会返回刚插入的自增列ID了,然后可以手工赋值到对象上面去 public static long Insert<T>(this IDbC ...
- Ext JS 6学习文档–第2章–核心概念
核心概念 在下一章我们会构建一个示例项目,而在这之前,你需要学习一些在 Ext JS 中的核心概念,这有助于你更容易理解示例项目.这一章我们将学习以下知识点: 类系统,创建和扩展类 事件 Ext JS ...
- Hadoop学习(一):完全分布式集群环境搭建
1. 设置免密登录 (1) 新建普通用户hadoop:useradd hadoop(2) 在主节点master上生成密钥对,执行命令ssh-keygen -t rsa便会在home文件夹下生成 .ss ...
- Python—字典(当索引不好用时)
一.定义与概念 1.字典是针对非序列集合而提供的一种数据类型 举例:检索学生信息. “<键><值>对”. 键(即身份证号码) 值(即学生信息). “键值对”例子 姓名和电话号码 ...
- POJ 2229 计数DP
dp[i]代表是数字i的最多组合数如果i是一个奇数,i的任意一个组合都包含1,所以dp[i] = dp[i-1] 如果i是一个偶数,分两种情况讨论,一种是序列中包含1,因此dp[i]=dp[i-1]一 ...
- 软工冲刺-Alpha 冲刺 (3/10)
队名:起床一起肝活队 组长博客:博客链接 作业博客:班级博客本次作业的链接 组员情况 组员1(队长):白晨曦 过去两天完成了哪些任务 描述: 很胖,刚学,照猫画虎做了登录与注册界面. 展示GitHub ...
- java—连连看GUI
1.连连看棋盘图形化 package Link; import java.awt.Color; import java.awt.Font; import java.awt.Graphics; impo ...