资讯专栏INFORMATION COLUMN

LeetCode 之 JavaScript 解答第二题 —— 两数相加(Add Two Number

Sunxb / 2747人阅读

摘要:多位数加多位数,反转链表转化整数,如果整数相加,可能会溢出,此方法行不通。直接进行位数运算,两链表每取出一个就做运算,将结果放入到新链表中。求和运算会出现额外的进位一般进位与最高位进位两种情况。两位数取模运算。

Time:2019/4/2
Title: ADD Two Numbers
Difficulty: medium
Author:小鹿
公众号:一个不甘平凡的码农。
题目二:ADD Two Numbers

You are given two non-empty linked lists representing two non-negative integers. The digits are stored in reverse order and each of their nodes contain a single digit. Add the two numbers and return it as a linked list.

You may assume the two numbers do not contain any leading zero, except the number 0 itself.

Example:

Input: (2 -> 4 -> 3) + (5 -> 6 -> 4)
Output: 7 -> 0 -> 8
Explanation: 342 + 465 = 807.

Solve:

▉ 算法思路:
1)观察 Example 规律,关联到链表,用一个带头的链表存储。

2)多位数加多位数,反转链表转化整数,如果整数相加,可能会溢出,此方法行不通。

3)直接进行位数运算,两链表每取出一个就做运算,将结果放入到新链表中。

▉ 临界条件:
1)一个链表比另一个链表长;

2)其中一个链表为 null。

3)求和运算会出现额外的进位(一般进位与最高位进位两种情况)。

▉ 步骤:
1)遍历链表之前,要定义一个哨兵结点、临时结点、存储计算结果的结点、进位标志;

2)开始遍历数据,判断当前结点是否为 null,为 null 就用 0 代替,否则取出数值;

3)求和(加 carray 进位),判断是否进位?记录进位值;

4)求模取余,计算两位数的各位数存储到链表中,指针向后移动;

5)判断结点是否为 null,继续遍历(如果链表 l2 比 l1 短,没有下一结点只能返回本身下次处理当做 null 处理)

6) 退出 while 循环勿忘最高位满位情况,carray 还存放着 1,所以判断最高位是否需要进位,存放到链表最后

▉ 代码实现:
/**
 * 性能分析:
 * 1)遍历整个链表,时间复杂度为 O(n)。
 * 2)需要额外的 n 大小的空间存储 计算结果结点,空间复杂度为 O(n)。
 */
var addTwoNumbers = function(l1, l2) {
    //定义哨兵结点
    let head = new ListNode("head");
    let current = head;//临时指针
    //存储计算后的链表
    let sumNode = head;
    //定义进位变量
    let carray = 0;
    //开始遍历两个链表取数据,判断链表是否为 null
    while(l1 !== null || l2 !== null){
        //判断取数据的链表是否为nulL,为 null 就用 0 替换
        let num1 = 0;
        let num2 = 0;
        if(l1 == null){
            num1 = 0;
        }else{
            num1 = l1.val;
        }
        if(l2 == null){
            num2 = 0;
        }else{
            num2 = l2.val;
        }
        // let num1 = l1 == null ? 0 : l1.val;
        // let num2 = l2 == null ? 0 : l2.val;
        //计算取出的两个数值的和用于判断是否满进位,如果满 10,carray 需要记录进位,默认为 0
        let sum = num1 + num2 + carray;
        //判断是否需要存储进位值 1
        if(sum > 9){
           carray = 1;
        }else{
            carray = 0;
        }
        //carray = sum > 9 ? 1 : 0;
        //将两数之和相加[取模(取余运算)]添加到 sumNode 新链表中,一次排列
        current.next = new ListNode(sum % 10)
        //将指针指向下一链表结点
        current = current.next;
        //继续遍历链表中的数据,判断下一结点是否为 null
        if(l1 !== null){
            l1 = l1.next;
        }else{
            //如果链表 l1 比 l2 短,没有下一结点只能返回本身下次处理当做 null 处理
            l1 = l1;
        }
        if(l2 !== null){
            l2 = l2.next;
        }else{
            //如果链表 l2 比 l1 短,没有下一结点只能返回本身下次处理当做 null 处理
            l2 = l2;
        }
        // l1 为不为 null 才满足条件
        // l1 = l1 ? l1.next : l1;
        // l2 = l2 ? l2.next : l2;
    }
    //最高位满位情况,carray 还存放着 1,所以判断最高位是否需要进位
    if(carray === 1){
        //有哨兵的,所以需要 next 才能存放下一结点
        current.next = new ListNode(1);
    }
    //返回哨兵结点之后的链表
    return head.next;
}
▉ 代码缩减:
var addTwoNumbers = function(l1, l2) {
    //定义哨兵结点
    let head = new ListNode("head");
    let current = head;//临时指针
    //存储计算后的链表
    let sumNode = head;
    //定义进位变量
    let carray = 0;
    //开始遍历两个链表取数据,判断链表是否为 null
    while(l1 !== null || l2 !== null){
        //判断取数据的链表是否为nulL,为 null 就用 0 替换
        let num1 = l1 == null ? 0 : l1.val;
        let num2 = l2 == null ? 0 : l2.val;
        //计算取出的两个数值的和用于判断是否满进位,如果满 10,carray 需要记录进位,默认为 0
        let sum = num1 + num2 + carray;
        //判断是否需要存储进位值 1
        if(sum > 9){
           carray = 1;
        }else{
            carray = 0;
        }
        //carray = sum > 9 ? 1 : 0;
        //将两数之和相加[取模(取余运算)]添加到 sumNode 新链表中,一次排列
        current.next = new ListNode(sum % 10)
        //将指针指向下一链表结点
        current = current.next;
        //继续遍历链表中的数据,判断下一结点是否为 null
        l1 为不为 null 才满足条件
        l1 = l1 ? l1.next : l1;
        l2 = l2 ? l2.next : l2;
    }
    //最高位满位情况,carray 还存放着 1,所以判断最高位是否需要进位
    if(carray === 1){
        //有哨兵的,所以需要 next 才能存放下一结点
        current.next = new ListNode(1);
    }
    //返回哨兵结点之后的链表
    return head.next;
}
▉ 总结:需要注意几点。
1、 l1 = l1 ? l1.next : l1 代表的是 l1 不等于 null 会去 l1.next 的值。

2、用到哨兵思想,所以注意当前的指针指向。

3、两位数取模运算。

▉ 扩展:
三位数怎么取得各个位置上的数字呢?(水仙花数)

答:

//移动小数点向前一位,得到小数点后一位
个位:a = 123 % 10 = 3
//移动小数点向前两位,得到小数点后两位,除以10取整
十位:b  = parseInt((123 % 100) / 10)
//移动小数点向前三位,得到小数点后三位,除以100取整
百位::c = parseInt((123 % 1000) / 100)
//依次类推.....

欢迎一起加入到 LeetCode 开源 Github 仓库,可以向 me 提交您其他语言的代码。在仓库上坚持和小伙伴们一起打卡,共同完善我们的开源小仓库!
Github:https://github.com/luxiangqia...


欢迎关注我个人公众号:「一个不甘平凡的码农」,记录了自己一路自学编程的故事。

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/103189.html

相关文章

  • LeetCode.2 两数相加(Add Two Numbers)(JS)

    摘要:更新之前说感觉优秀答案的最后三行可以用尾递归优化不知道尾递归的小伙伴可以点这里,仔细想了一下,并不能。尾递归的实现,往往需要改写递归函数,确保最后一步只调用自身。 上周日就想写vue.nextTick的源码分析,可是总是不知道从哪儿下手,今天有时间,先把leetcode第二题补了,感觉这道题还挺简单的 一、题目 两数相加: 给出两个 非空 的链表用来表示两个非负的整数。其中,它们各自...

    Binguner 评论0 收藏0
  • LeetCode 2:两数相加 Add Two Numbers

    摘要:给出两个非空的链表用来表示两个非负的整数。如果,我们将这两个数相加起来,则会返回一个新的链表来表示它们的和。需要考虑到两个链表长度不同时遍历方式链表遍历完成时最后一位是否需要进一位。 ​给出两个 非空 的链表用来表示两个非负的整数。其中,它们各自的位数是按照 逆序 的方式存储的,并且它们的每个节点只能存储 一位 数字。如果,我们将这两个数相加起来,则会返回一个新的链表来表示它们的和。 ...

    diabloneo 评论0 收藏0
  • LeetCode 2:两数相加 Add Two Numbers

    摘要:给出两个非空的链表用来表示两个非负的整数。如果,我们将这两个数相加起来,则会返回一个新的链表来表示它们的和。需要考虑到两个链表长度不同时遍历方式链表遍历完成时最后一位是否需要进一位。 ​给出两个 非空 的链表用来表示两个非负的整数。其中,它们各自的位数是按照 逆序 的方式存储的,并且它们的每个节点只能存储 一位 数字。如果,我们将这两个数相加起来,则会返回一个新的链表来表示它们的和。 ...

    Towers 评论0 收藏0
  • LeetCode JavaScript 解答第一题 —— 两数和(Two Sum)

    摘要:步骤遍历数组数据,将根据下标和元素值存放到散列表中。目标值减去数组元素差值并在散列表中查找。测试法三一遍哈希表算法思路遍历目标值减去数组元素的差值同时判断该值在散列表中是否存在差值,如果存在,则返回否则将数据加入到散列表中。 Time:2019/4/1Title:Two SumDifficulty: simpleAuthor:小鹿 题目一:Two Sum Given an array ...

    k00baa 评论0 收藏0
  • LeetCode 攻略 - 2019 年 7 月下半月汇总(100 题攻略)

    摘要:月下半旬攻略道题,目前已攻略题。目前简单难度攻略已经到题,所以后面会调整自己,在刷算法与数据结构的同时,攻略中等难度的题目。 Create by jsliang on 2019-07-30 16:15:37 Recently revised in 2019-07-30 17:04:20 7 月下半旬攻略 45 道题,目前已攻略 100 题。 一 目录 不折腾的前端,和咸鱼有什么区别...

    tain335 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<