LeetCode 2 - 两数相加

题目描述
给出两个 非空 的链表用来表示两个非负的整数。其中,它们各自的位数是按照 逆序 的方式存储的,并且它们的每个节点只能存储 一位 数字。
如果,我们将这两个数相加起来,则会返回一个新的链表来表示它们的和。
您可以假设除了数字 0 之外,这两个数都不会以 0 开头。
示例:
输入:(2 -> 4 -> 3) + (5 -> 6 -> 4)
输出:7 -> 0 -> 8
原因:342 + 465 = 807

解法一:Python
把链表转化为数字进行求和后在返回成链表,当然这道题有些处心积虑的是如果用C/C++这个方法就不得行,因为其中有测试样例超出了int的长度,但是对Python来说可以全然不顾。当然这个方法也是最为简便的方法,既不用考虑进位,也不用担心两数字长度不一要补0.

# Definition for singly-linked list.
# class ListNode:
#     def __init__(self, x):
#         self.val = x
#         self.next = None

class Solution:
    def addTwoNumbers(self, l1: ListNode, l2: ListNode) -> ListNode:
        def get_num(l):
            if not l: return 0
            return l.val + get_num(l.next) * 10
        def make_list(n):
            if n == 0: return None
            l = ListNode(n%10)
            l.next = make_list(n//10)
            return l
        s = get_num(l1) + get_num(l2)
        if s == 0:
            return l1
        return make_list(s)

注意,在Python中," / “就表示 浮点数除法,返回浮点结果;” // "表示整数除法。

解法二:C++ 利用链表,原理同大数加法
伪代码如下:


  • 将当前结点初始化为返回列表的哑结点
  • 将进位 carry 初始化为 0
  • 将 p 和 q 分别初始化为列表 l1 和 l2 的头部
  • 遍历列表 l1 和 l2 直至到达它们的尾端
      · 将 x 设为结点 p 的值。如果 p 已经到达 l1 的末尾,则将其值设置为 0
      · 将 y 设为结点 q 的值。如果 q 已经到达 l2 的末尾,则将其值设置为 0
      · 设定 sum = x + y + carry
      · 更新进位的值,carry = sum / 10
      · 创建一个数值为 (summod10) 的新结点,并将其设置为当前结点的下一个结点,然后将当前结点前进到下一个结点
      ·同时,将 p 和 q 前进到下一个结点
  • 检查 carry = 1c是否成立,如果成立,则向返回列表追加一个含有数字 1 的新结点。
  • 返回哑结点的下一个结点

请注意,我们使用哑结点来简化代码。如果没有哑结点,则必须编写额外的条件语句来初始化表头的值

请特别注意以下情况:
l1=[0,1],l2=[0,1,2] 当一个列表比另一个列表长时
l1=[],l2=[0,1] 当一个列表为空时,即出现空列表
l1=[9,9],l2=[1] 求和运算最后可能出现额外的进位,这一点很容易被遗忘

/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode(int x) : val(x), next(NULL) {}
 * };
 */
class Solution {
public:
    ListNode* addTwoNumbers(ListNode* l1, ListNode* l2) {
        ListNode dummyHead(0), *curr = &dummyHead; 
        ListNode *p = l1, *q = l2;
        int carry = 0;
        while(p!=NULL||q!=NULL)
        {
            int x = (p!=NULL) ? p->val:0;
            int y = (q!=NULL) ? q->val:0;
            int sum = carry +x + y;
            carry = sum / 10;
            curr->next = new ListNode(sum%10);
            curr = curr->next;
            if(p!=NULL) p = p->next;
            if(q!=NULL) q = q->next;
        }
        if(carry>0) curr->next = new ListNode(carry);
        return dummyHead.next;
    }
};

复杂度分析

时间复杂度 O ( max ⁡ ( m , n ) ) O(\max(m, n)) O(max(m,n)),假设 m m m n n n 分别表示 l 1 l1 l1 l 2 l2 l2 的长度,上面的算法最多重复 max ⁡ ( m , n ) \max(m, n) max(m,n) 次。
空间复杂度 O ( max ⁡ ( m , n ) ) O(\max(m, n)) O(max(m,n)), 新列表的长度最多为 max ⁡ ( m , n ) + 1 \max(m,n) + 1 max(m,n)+1

这里补充说明几个知识:

  1. *p = &head 和 *p = list:前者是在申明的时候取head的地址,后者是指向list的指针
  2. " ->" 和 “.” 的区别:前者是指针,后者是结构体
  3. NULL和nullptr的区别:参见c++中nullptr
根据提供的引用内容,Leetcode 2 "两数相加"是一个涉及链表的问题。该问题给定了两个非负整数,每个整数的每一位都是按照逆序的方式存储在链表中。我们需要将这两个链表相加,并返回一个新的链表作为结果。 具体解题思路可以使用迭代法或递归法来解决。迭代法的伪代码如下所示: ``` 初始化一个哑节点 dummy 和一个进位 carry,同时把两个链表的头节点分别赋值给 p 和 q 遍历链表,直到 p 和 q 都为 None 计算当前的和 sum 为 p.val + q.val + carry 计算当前的进位 carry 为 sum // 10 创建一个新节点 node,节点的值为 sum % 10 把新节点连接到结果链表的尾部 更新 p 和 q 分别为 p.next 和 q.next 如果最后还有进位 carry,则创建一个新节点 node,节点的值为 carry,并连接到结果链表的尾部 返回结果链表的头节点 dummy.next ``` 递归法的伪代码如下所示: ``` 定义一个辅助函数 addTwoNumbersHelper,输入为两个链表的头节点 p 和 q,以及进位 carry 如果 p 和 q 都为 None 且 进位 carry 为 0,则返回 None 计算当前的和 sum 为 p.val + q.val + carry 计算当前的进位 carry 为 sum // 10 创建一个新节点 node,节点的值为 sum % 10 设置新节点的下一个节点为递归调用 addTwoNumbersHelper(p.next, q.next, carry) 返回新节点 返回 addTwoNumbersHelper(p, q, 0) 的结果 以上是解决 Leetcode 2 "两数相加"问题的两种方法。如果你还有其他相关问题,请
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值