LEETCODE(力扣)2. 两数相加
给你两个 非空 的链表,表示两个非负的整数。它们每位数字都是按照 逆序 的方式存储的,并且每个节点只能存储 一位 数字。
请你将两个数相加,并以相同形式返回一个表示和的链表。
你可以假设除了数字 0 之外,这两个数都不会以 0 开头。
示例 1:
输入:l1 = [2,4,3], l2 = [5,6,4]
输出:[7,0,8]
解释:342 + 465 = 807.
示例 2:
输入:l1 = [0], l2 = [0]
输出:[0]
示例 3:
输入:l1 = [9,9,9,9,9,9,9], l2 = [9,9,9,9]
输出:[8,9,9,9,0,0,0,1]
提示:
每个链表中的节点数在范围 [1, 100] 内
0 <= Node.val <= 9
题目数据保证列表表示的数字不含前导零
将两节点相加后的和同时赋给两节点,直到一个链表被遍历完全,此时继续用进位标志位遍历剩下的那个较长的链表,然后返回该链表。

从时间与空间上来看找不到问题,但问题在于:
1.该方法对原本的链表进行了破环
2.逻辑虽简单,但代码量较长
自解
/**
* Definition for singly-linked list.
* struct ListNode {
* int val;
* struct ListNode *next;
* };
*/
struct ListNode jinwei;
struct ListNode* addTwoNumbers(struct ListNode* l1, struct ListNode* l2) {
int a=0;
struct ListNode *p1,*p2;
p1=l1;
p2=l2;
while(p1->next!=NULL&&p2->next!=NULL)
{
if(a)
{
p1->val=p1->val+p2->val+a;
a=0;
if(p1->val>=10)
{
p1->val=p1->val%10;
a=1;
}
p2->val=p1->val;
p1=p1->next;
p2=p2->next;
continue;
}
p1->val=p1->val+p2->val;
if(p1->val>=10)
{
p1->val=p1->val%10;
a=1;
}
p2->val=p1->val;
p1=p1->next;
p2=p2->next;
}
if(p1->next==NULL&&p2->next!=NULL)
{
p2->val=p1->val+p2->val+a;
if(p2->val>=10)
{
p2->val=p2->val%10;
a=1;
}
else
{
a=0;
}
p2=p2->next;
while(a)
{
p2->val=p2->val+a;
if(p2->val>=10)
{
p2->val=p2->val%10;
a=1;
}
else
{
a=0;
}
if(p2->next!=NULL)
{
p2=p2->next;
}
else
{
if(a)
{
p2->next=&jinwei;
jinwei.val=1;
jinwei.next=NULL;
a=0;
}
}
}
return l2;
}
else if(p2->next==NULL&&p1->next!=NULL)
{
p1->val=p2->val+p1->val+a;
if(p1->val>=10)
{
p1->val=p1->val%10;
a=1;
}
else
{
a=0;
}
p1=p1->next;
while(a)
{
p1->val=p1->val+a;
if(p1->val>=10)
{
p1->val=p1->val%10;
a=1;
}
else
{
a=0;
}
if(p1->next!=NULL)
{
p1=p1->next;
}
else
{
if(a)
{
p1->next=&jinwei;
jinwei.val=1;
jinwei.next=NULL;
a=0;
}
}
}
return l1;
}
else
{
p1->val=p2->val+p1->val+a;
if(p1->val>=10)
{
p1->val=p1->val%10;
a=1;
}
else
{
a=0;
}
if(a)
{
p1->next=&jinwei;
jinwei.val=1;
}
return l1;
}
}
力扣解:
该解为两链表的结果申请了单独的空间进行储存,保留了原链表的完整性
struct ListNode* addTwoNumbers(struct ListNode* l1, struct ListNode* l2) {
struct ListNode dummy; // 哨兵节点
struct ListNode* cur = &dummy;
int carry = 0; // 进位
while (l1 || l2 || carry) { // 有一个不是空节点,或者还有进位,就继续迭代
if (l1) {
carry += l1->val; // 节点值和进位加在一起
l1 = l1->next; // 下一个节点
}
if (l2) {
carry += l2->val; // 节点值和进位加在一起
l2 = l2->next; // 下一个节点
}
cur = cur->next = malloc(sizeof(struct ListNode));
cur->val = carry % 10; // 每个节点保存一个数位
carry /= 10; // 新的进位
}
cur->next = NULL; // 注意最后一个节点是 malloc 出来的,next 不一定是 NULL,需要手动置为 NULL
return dummy.next; // 哨兵节点的下一个节点就是头节点
}
作者:灵茶山艾府
链接:https://leetcode.cn/problems/add-two-numbers/solutions/2327008/dong-hua-jian-ji-xie-fa-cong-di-gui-dao-oe0di/
来源:力扣(LeetCode)
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
仿写
自己参考了上方解答后尝试仿写
可以看出逻辑相比自解更简单明了,虽然空间上变差了,但这是因为单独为结果申请了空间,该解更具有实用性
/**
* Definition for singly-linked list.
* struct ListNode {
* int val;
* struct ListNode *next;
* };
*/
struct ListNode* addTwoNumbers(struct ListNode* l1, struct ListNode* l2) {
struct ListNode dummy,*p=&dummy;
int jinweiF=0;
while(l1||l2||jinweiF)
{
struct ListNode *temp=(struct ListNode *)malloc(sizeof(struct ListNode));
temp->next=NULL;
temp->val=0;
if(l1!=NULL){temp->val+=l1->val; l1=l1->next;}
if(l2!=NULL){temp->val+=l2->val; l2=l2->next;}
temp->val+=jinweiF;
if(temp->val>=10)
{
temp->val%=10;
jinweiF=1;
}else jinweiF=0;
p->next=temp;
p=p->next;
}
return dummy.next;
}

浙公网安备 33010602011771号