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
题目数据保证列表表示的数字不含前导零

将两节点相加后的和同时赋给两节点,直到一个链表被遍历完全,此时继续用进位标志位遍历剩下的那个较长的链表,然后返回该链表。
image
从时间与空间上来看找不到问题,但问题在于:
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;
}
posted @ 2025-04-27 14:32  Osen  阅读(31)  评论(0)    收藏  举报