LeetCode 之 JavaScript 解答第二題 —— 兩數相加(Add Two Numbers)

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.javascript

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

Example:node

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

Solve:git

▉ 算法思路:
1)觀察 Example 規律,關聯到鏈表,用一個帶頭的鏈表存儲。

2)多位數加多位數,反轉鏈表轉化整數,若是整數相加,可能會溢出,此方法行不通。github

3)直接進行位數運算,兩鏈表每取出一個就作運算,將結果放入到新鏈表中。算法

▉ 臨界條件:
1)一個鏈表比另外一個鏈表長;

2)其中一個鏈表爲 null。編程

3)求和運算會出現額外的進位(通常進位與最高位進位兩種狀況)。性能

▉ 步驟:
1)遍歷鏈表以前,要定義一個哨兵結點、臨時結點、存儲計算結果的結點、進位標誌;

2)開始遍歷數據,判斷當前結點是否爲 null,爲 null 就用 0 代替,不然取出數值;指針

3)求和(加 carray 進位),判斷是否進位?記錄進位值;code

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;
}
▉ 總結:須要注意幾點。
一、 l1 = l1 ? l1.next : l1 表明的是 l1 不等於 null 會去 l1.next 的值。

二、用到哨兵思想,因此注意當前的指針指向。

三、兩位數取模運算。

▉ 擴展:
三位數怎麼取得各個位置上的數字呢?(水仙花數)

答:

//移動小數點向前一位,獲得小數點後一位
個位:a = 123 % 10 = 3
//移動小數點向前兩位,獲得小數點後兩位,除以10取整
十位:b  = parseInt((123 % 100) / 10)
//移動小數點向前三位,獲得小數點後三位,除以100取整
百位::c = parseInt((123 % 1000) / 100)
//依次類推.....

歡迎一塊兒加入到 LeetCode 開源 Github 倉庫,能夠向 me 提交您其餘語言的代碼。在倉庫上堅持和小夥伴們一塊兒打卡,共同完善咱們的開源小倉庫!
Github:https://github.com/luxiangqia...

歡迎關注我我的公衆號:「一個不甘平凡的碼農」,記錄了本身一路自學編程的故事。
相關文章
相關標籤/搜索