成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

LeetCode.2 兩數(shù)相加(Add Two Numbers)(JS)

Binguner / 1816人閱讀

摘要:更新之前說感覺優(yōu)秀答案的最后三行可以用尾遞歸優(yōu)化不知道尾遞歸的小伙伴可以點(diǎn)這里,仔細(xì)想了一下,并不能。尾遞歸的實(shí)現(xiàn),往往需要改寫遞歸函數(shù),確保最后一步只調(diào)用自身。

上周日就想寫vue.nextTick的源碼分析,可是總是不知道從哪兒下手,今天有時(shí)間,先把leetcode第二題補(bǔ)了,感覺這道題還挺簡(jiǎn)單的
一、題目

兩數(shù)相加:

給出兩個(gè) 非空 的鏈表用來表示兩個(gè)非負(fù)的整數(shù)。其中,它們各自的位數(shù)是按照 逆序 的方式存儲(chǔ)的,并且它們的每個(gè)節(jié)點(diǎn)只能存儲(chǔ) 一位 數(shù)字。如果,我們將這兩個(gè)數(shù)相加起來,則會(huì)返回一個(gè)新的鏈表來表示它們的和。您可以假設(shè)除了數(shù)字 0 之外,這兩個(gè)數(shù)都不會(huì)以 0 開頭。
示例:

示例

輸入:(2 -> 4 -> 3) + (5 -> 6 -> 4)
輸出:7 -> 0 -> 8
原因:342 + 465 = 807

二、我的答案
/**
 *. Definition for singly-linked list.
 *. function ListNode(val) {
 *.     this.val = val;
 *.     this.next = null;
 *. }
 */
/**
 *. @param {ListNode} l1
 *. @param {ListNode} l2
 *. @return {ListNode}
 */
var addTwoNumbers = function(l1, l2) { 
      function ListToArray(list) {
        if(list.next) {
          return [list.val, ...ListToArray(list.next)]
        } else {
          return [list.val]
        }
      }
      function sumArray(arr1, arr2) {
        if(arr1.length < arr2.length) {
          let arr = []
          arr = arr1
          arr1 = arr2
          arr2 = arr
        }
        let littleLen = arr2.length
        let i =0
        for(; i < littleLen; i++) {
          arr1[i] += arr2[i]
          if(arr1[i] >= 10) {
            arr1[i] -= 10
            arr1[i + 1] ? arr1[i + 1]++ : arr1[i+1] =1
          }
        }
        for(; i < arr1.length; i++ ) {
          if(arr1[i] >= 10) {
            arr1[i] -= 10
            arr1[i + 1] ? arr1[i + 1]++ : arr1[i+1] =1
          }
        }
        return arr1.reverse()
      }

      function ArrayToList(arr) {
        if(arr.length > 0) {
          let node = new ListNode(arr.pop())
          node.next = ArrayToList(arr)
          return node
        } else {
          return null
        }
      }
      return ArrayToList(sumArray(ListToArray(l1),ListToArray(l2)))
    };

計(jì)算兩個(gè)鏈表表示的數(shù)的和,統(tǒng)共分三步:

把冰箱門打開 鏈表轉(zhuǎn)數(shù)組;

兩個(gè)數(shù)組相加得到和數(shù)組;

和數(shù)組轉(zhuǎn)鏈表

我的寫法答案簡(jiǎn)單易懂,就不做解釋,這里說一下寫的時(shí)候碰到一個(gè)坑
關(guān)于第二步兩個(gè)數(shù)組相加,一開始的思路并不是這樣的,而是轉(zhuǎn)成String再轉(zhuǎn)Number相加再轉(zhuǎn)回來,那么這個(gè)思路折哪兒了呢,有一個(gè)測(cè)試用例是[1,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,1]和[5,6,4],轉(zhuǎn)成Number相加得1e+30,沒錯(cuò),萬(wàn)惡的弱類型,查詢能不能不轉(zhuǎn)換,未果,就只能遍歷各數(shù)位相加了

這題一開始沒懂啥意思,懂了之后思路還挺清晰的,擊敗33.45%,嘛,喜聞樂見,看看大手子們是咋寫的吧

三、優(yōu)秀答案
/**
 * Definition for singly-linked list.
 * function ListNode(val) {
 *     this.val = val;
 *     this.next = null;
 * }
 */
/**
 * @param {ListNode} l1
 * @param {ListNode} l2
 * @return {ListNode}
 */
var addTwoNumbers = function(l1, l2, curr = 0) {
    if(l1 === null && l2 === null) {
        if(curr) return new ListNode(curr)
        else return null;
    } else {
        if(l1 == null) l1 = new ListNode(0)
        if(l2 == null) l2 = new ListNode(0)
        let nextVal = (l2.val || 0) + (l1.val || 0) + curr
        curr = 0
        if(nextVal > 9) {
            curr = 1
            nextVal -= 10
        }
        l1.val = nextVal
        l1.next = addTwoNumbers(l1.next, l2.next, curr)
        return l1
    }
}; 

這個(gè)是我看到的最優(yōu)秀的答案,本地跑擊敗91%,甩我不知道幾條街,看各種優(yōu)秀答案總是能刷新認(rèn)知,來看他的思路
首先他把函數(shù)改了,函數(shù)的本意是相加兩個(gè)鏈表,他改成了相加兩個(gè)節(jié)點(diǎn),參數(shù)中curr的意思是上一組節(jié)點(diǎn)相加是否進(jìn)位這什么命名
再來看函數(shù)內(nèi),if部分的代碼意為處理最后的邊界情況,例[1],[9,9,9,9],即遍歷完兩個(gè)鏈表之后,如果上一次遍歷有進(jìn)位,則new一個(gè)節(jié)點(diǎn)
else內(nèi)則是主要的節(jié)點(diǎn)相加部分,其中

    if(l1 == null) l1 = new ListNode(0)
    if(l2 == null) l2 = new ListNode(0)
    let nextVal = (l2.val || 0) + (l1.val || 0) + curr

(l2.val || 0)明顯是多余的,我猜他本來寫的是let nextVal = (l2.val || 0) + (l1.val || 0) + curr但是執(zhí)行發(fā)現(xiàn)null點(diǎn)不出val,就加了上面的判斷不過下面的沒刪,不精簡(jiǎn)代碼差評(píng)。
剩下的邏輯就顯而易見了,進(jìn)位什么的,不過這種同步遍歷真的很巧妙,我怎么就是學(xué)不會(huì)/捂臉

四、路漫漫其修遠(yuǎn)兮

???????這幾道題提交都是在力扣上的,但是做完這道題去看優(yōu)秀答案,發(fā)現(xiàn)有一個(gè)排名較前的答案返回值是Array,我自己復(fù)制過來甚至根本過不了測(cè)試用例,應(yīng)該是力扣沒有像leetcode更新全面吧,題改了但是答案卻沒有改,決定以后還是提交到leetCode吧,
???????再細(xì)看優(yōu)秀答案的最后三行,怎么總感覺好像可以用尾調(diào)用優(yōu)化一下,有空看一下,今天這篇博客到此為止,下面該做的第四題難度為hard,害怕之余突然有點(diǎn)期待看到各種神仙解法/捂臉。

THE END

2019.3.27更新
之前說感覺優(yōu)秀答案的最后三行可以用尾遞歸優(yōu)化(不知道尾遞歸的小伙伴可以點(diǎn)這里),仔細(xì)想了一下,并不能。

尾遞歸的實(shí)現(xiàn),往往需要改寫遞歸函數(shù),確保最后一步只調(diào)用自身。做到這一點(diǎn)的方法,就是把所有用到的內(nèi)部變量改寫成函數(shù)的參數(shù)。

原因如下:
????????假設(shè)我們按照上面描述實(shí)現(xiàn)尾遞歸,那么函數(shù)需要三個(gè)參數(shù),分別是l1, l2,還有當(dāng)前組裝的listNode對(duì)象,那么我們每次調(diào)用的時(shí)候,都需要l1.val + l2.val賦值給當(dāng)前組裝的對(duì)象的最深層,要想獲得對(duì)象的最深層,就得遍歷,那我們還優(yōu)化個(gè)錘子。
????????不過這么分析下來,優(yōu)秀答案作者可能本來的意圖就是用尾遞歸優(yōu)化,所以給第三個(gè)參數(shù)命名為curr,發(fā)現(xiàn)得不償失后放棄這種做法,把第三個(gè)參數(shù)的作用改為進(jìn)位,但是并沒有再修改變量名,不過這個(gè)編碼習(xí)慣倒是很符合上文我猜測(cè)的let nextVal = (l2.val || 0) + (l1.val || 0) + curr這行代碼沒有精簡(jiǎn)的理由。
????????分析個(gè)優(yōu)秀答案能分析出破案的感覺我也是服我自己

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/103021.html

相關(guān)文章

  • Leetcode 2 Add Two Numbers 兩數(shù)相加

    摘要:這題是說給出兩個(gè)鏈表每個(gè)鏈表代表一個(gè)多位整數(shù)個(gè)位在前比如代表著求這兩個(gè)鏈表代表的整數(shù)之和同樣以倒序的鏈表表示難度這個(gè)題目就是模擬人手算加法的過程需要記錄進(jìn)位每次把對(duì)應(yīng)位置兩個(gè)節(jié)點(diǎn)如果一個(gè)走到頭了就只算其中一個(gè)的值加上進(jìn)位值 Add Two Numbers You are given two linked lists representing two non-negative num...

    Charlie_Jade 評(píng)論0 收藏0
  • LeetCode 2兩數(shù)相加 Add Two Numbers

    摘要:給出兩個(gè)非空的鏈表用來表示兩個(gè)非負(fù)的整數(shù)。如果,我們將這兩個(gè)數(shù)相加起來,則會(huì)返回一個(gè)新的鏈表來表示它們的和。需要考慮到兩個(gè)鏈表長(zhǎng)度不同時(shí)遍歷方式鏈表遍歷完成時(shí)最后一位是否需要進(jìn)一位。 ?給出兩個(gè) 非空 的鏈表用來表示兩個(gè)非負(fù)的整數(shù)。其中,它們各自的位數(shù)是按照 逆序 的方式存儲(chǔ)的,并且它們的每個(gè)節(jié)點(diǎn)只能存儲(chǔ) 一位 數(shù)字。如果,我們將這兩個(gè)數(shù)相加起來,則會(huì)返回一個(gè)新的鏈表來表示它們的和。 ...

    diabloneo 評(píng)論0 收藏0
  • LeetCode 2兩數(shù)相加 Add Two Numbers

    摘要:給出兩個(gè)非空的鏈表用來表示兩個(gè)非負(fù)的整數(shù)。如果,我們將這兩個(gè)數(shù)相加起來,則會(huì)返回一個(gè)新的鏈表來表示它們的和。需要考慮到兩個(gè)鏈表長(zhǎng)度不同時(shí)遍歷方式鏈表遍歷完成時(shí)最后一位是否需要進(jìn)一位。 ?給出兩個(gè) 非空 的鏈表用來表示兩個(gè)非負(fù)的整數(shù)。其中,它們各自的位數(shù)是按照 逆序 的方式存儲(chǔ)的,并且它們的每個(gè)節(jié)點(diǎn)只能存儲(chǔ) 一位 數(shù)字。如果,我們將這兩個(gè)數(shù)相加起來,則會(huì)返回一個(gè)新的鏈表來表示它們的和。 ...

    Towers 評(píng)論0 收藏0
  • LeetCode 之 JavaScript 解答第二題 —— 兩數(shù)相加Add Two Number

    摘要:多位數(shù)加多位數(shù),反轉(zhuǎn)鏈表轉(zhuǎn)化整數(shù),如果整數(shù)相加,可能會(huì)溢出,此方法行不通。直接進(jìn)行位數(shù)運(yùn)算,兩鏈表每取出一個(gè)就做運(yùn)算,將結(jié)果放入到新鏈表中。求和運(yùn)算會(huì)出現(xiàn)額外的進(jìn)位一般進(jìn)位與最高位進(jìn)位兩種情況。兩位數(shù)取模運(yùn)算。 Time:2019/4/2Title: ADD Two NumbersDifficulty: mediumAuthor:小鹿公眾號(hào):一個(gè)不甘平凡的碼農(nóng)。 題目二:ADD Two...

    Sunxb 評(píng)論0 收藏0
  • LeetCode 167:兩數(shù)之和 II - 輸入有序數(shù)組 Two Sum II - Input a

    摘要:公眾號(hào)愛寫給定一個(gè)已按照升序排列的有序數(shù)組,找到兩個(gè)數(shù)使得它們相加之和等于目標(biāo)數(shù)。函數(shù)應(yīng)該返回這兩個(gè)下標(biāo)值和,其中必須小于。示例輸入輸出解釋與之和等于目標(biāo)數(shù)。 公眾號(hào): 愛寫bug(ID:icodebugs) 給定一個(gè)已按照升序排列 的有序數(shù)組,找到兩個(gè)數(shù)使得它們相加之和等于目標(biāo)數(shù)。 函數(shù)應(yīng)該返回這兩個(gè)下標(biāo)值 index1 和 index2,其中 index1 必須小于 index2。...

    張春雷 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

閱讀需要支付1元查看
<