摘要:首先是鏈表的定義語法搞錯了。分析本題與編程之美上的從無頭單鏈表中刪除節(jié)點(diǎn)類似。但是如果節(jié)點(diǎn)是尾節(jié)點(diǎn)時,該方法就行不通了。分析非遞歸的算法很簡單,用三個臨時指針在鏈表上循環(huán)一遍即可。遞歸算法是先逆轉(zhuǎn)下一個節(jié)點(diǎn),再逆轉(zhuǎn)當(dāng)前節(jié)點(diǎn)。
鏈接描述## 面試前準(zhǔn)備了Promise的一種實(shí)現(xiàn)(大致理解和寫出來),二叉樹的構(gòu)建,刪除,查找,插入,快排的非遞歸,準(zhǔn)備了蠻多的吧,但是沒考慮鏈表。然后考個鏈表的反轉(zhuǎn),其實(shí)很簡單,只不過有點(diǎn)懵,比我想的稍微難了點(diǎn),然后這種考慮算法的實(shí)現(xiàn)而忽略了特殊情況的考慮,雖然我確定我能寫出來,但是在那段懵的時間沒有。后來看到了我朋友寫的鏈表面試題,鏈接在此。
我朋友是做Android開發(fā)的,以java的形式舉了幾個例子。我是眼高手低,還是練練javascript的形式吧。
首先是鏈表的定義
/*class Node{ //語法搞錯了。
constructor(val){
this.val= val;
this.next = null;
}
}*/
function Node(val){
this.val = val;
this.next = null;
}
題目描述:給定單鏈表中需要刪除的節(jié)點(diǎn)(不是尾節(jié)點(diǎn)),在 O(1) 時間刪除該節(jié)點(diǎn)。
分析:本題與《編程之美》上的「從無頭單鏈表中刪除節(jié)點(diǎn)」類似。主要思想都是「貍貓換太子」,即用下一個節(jié)點(diǎn)數(shù)據(jù)覆蓋要刪除的節(jié)點(diǎn),然后刪除下一個節(jié)點(diǎn)。但是如果節(jié)點(diǎn)是尾節(jié)點(diǎn)時,該方法就行不通了。
function deleteNode(node){ if(node==null||node.next==null) return ; node.val = node.next.val; node.next = node.next.next; }
題目描述:輸出一個單鏈表的逆序反轉(zhuǎn)后的鏈表。
分析:非遞歸的算法很簡單,用三個臨時指針 prev、cur、next 在鏈表上循環(huán)一遍即可。遞歸算法是先逆轉(zhuǎn)下一個節(jié)點(diǎn),再逆轉(zhuǎn)當(dāng)前節(jié)點(diǎn)。
有點(diǎn)傷心的題目啊,其實(shí)還是蠻簡單的,就是在改變next的指向的時候,保留next指向的節(jié)點(diǎn)。
function reverseByLoop(head){ if(head == null || head.next == null) return head; var pre = null, next; while(head!=null){ next = head.next; head.next = pre; pre = head; head = next; } return pre; } function reverseByRecursion(head){ if(head==null||head.next ==null) return head; //遞歸的好難理解。。 var headOfNext = reverseByRecursion(head.next); head.next.next = head; head.next = null; return headOfNext; }
遞歸有一種給我一個支點(diǎn)和足夠長的棍,我就能翹起地球一樣的感覺。
比如原鏈表是這樣的結(jié)構(gòu)A->B->C->D->E.
既然這個function的目的是把鏈表倒序,那么如果我調(diào)用reverseByRecursion(B),
那么返回的結(jié)果一定是E,并且E->D->C->B.
現(xiàn)在我手中還有A,A->B。所以為了形成E->D->C->B->A, 只需要把B指向A,然后A指向NULL. 所以就是后續(xù)操作了。
那么還需要驗(yàn)證臨界情況,即最里面一層執(zhí)行的狀態(tài)能否達(dá)成逆轉(zhuǎn)鏈表的效果。
調(diào)用棧如下
reverseByRecursion(E)->return E;
reverseByRecursion(D)-> D.next.next=D;=>E->D D.next=NULL; return E; E->D->NULL
reverseByRecursion(C)-> C.next.next=C;=>D.next=C C.next=NULL; return E;=>E->D->C->NULL
reverseByRecursion(B)
reverseByRecursion(A)
先這樣吧。果然有點(diǎn)眼高手低。
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/70691.html
摘要:強(qiáng)烈推薦上值得前端學(xué)習(xí)的數(shù)據(jù)結(jié)構(gòu)與算法項(xiàng)目,包含圖的演示過程與視頻講解。該倉庫包含了多種基于的算法與數(shù)據(jù)結(jié)構(gòu),提供進(jìn)一步閱讀的解釋和鏈接。數(shù)據(jù)結(jié)構(gòu)和算法必知必會的個代碼實(shí)現(xiàn)。 showImg(https://segmentfault.com/img/bVbvpYZ); 前言 算法為王。想學(xué)好前端,先練好內(nèi)功,內(nèi)功不行,就算招式練的再花哨,終究成不了高手;只有內(nèi)功深厚者,前端之路才會走得...
摘要:摘要是如何回收內(nèi)存的深入淺出系列深入淺出第課箭頭函數(shù)中的究竟是什么鬼深入淺出第課函數(shù)是一等公民是什么意思呢深入淺出第課什么是垃圾回收算法最近垃圾回收這個話題非?;穑蠹也荒茈S隨便便的扔垃圾了,還得先分類,這樣方便對垃圾進(jìn)行回收再利用。 摘要: JS是如何回收內(nèi)存的? 《JavaScript深入淺出》系列: JavaScript深入淺出第1課:箭頭函數(shù)中的this究竟是什么鬼? Jav...
摘要:代碼實(shí)現(xiàn)六堆排序算法簡介堆排序是指利用堆這種數(shù)據(jù)結(jié)構(gòu)所設(shè)計的一種排序算法。九計數(shù)排序算法簡介計數(shù)排序是一種穩(wěn)定的排序算法。計數(shù)排序不是比較排序,排序的速度快于任何比較排序算法。 贊助我以寫出更好的文章,give me a cup of coffee? 2017最新最全前端面試題 1、插入排序 1)算法簡介 插入排序(Insertion-Sort)的算法描述是一種簡單直觀的排序算法。它...
摘要:快速排序是不穩(wěn)定的排序算法。瀏覽器的實(shí)現(xiàn)不同有什么影響排序算法不穩(wěn)定有什么影響舉個例子某市的機(jī)動車牌照拍賣系統(tǒng),最終中標(biāo)的規(guī)則為按價格進(jìn)行倒排序相同價格則按照競標(biāo)順位即價格提交時間進(jìn)行正排序。 本文要解決的問題 1、找出 Array.prototype.sort 使用的什么排序算法 2、用一種直觀的方式展示 Array.prototype.sort 的時間復(fù)雜度,看看它有多快? 3、...
摘要:算法系列單源最短路徑算法迪杰斯特拉算法是由荷蘭計算機(jī)科學(xué)家狄克斯特拉于年提出的,因此又叫狄克斯特拉算法。是從一個頂點(diǎn)到其余各頂點(diǎn)的最短路徑算法,解決的是有向圖中最短路徑問題。 Javascript算法系列 - 單源最短路徑 - Dijkstra算法 迪杰斯特拉算法是由荷蘭計算機(jī)科學(xué)家狄克斯特拉于1959年提出的,因此又叫狄克斯特拉算法。是從一個頂點(diǎn)到其余各頂點(diǎn)的最短路徑算法,解決的是有...
摘要:常見的內(nèi)部排序算法有插入排序希爾排序選擇排序冒泡排序歸并排序快速排序堆排序基數(shù)排序等。用一張圖概括歸并排序英語,或,是創(chuàng)建在歸并操作上的一種有效的排序算法,效率為。 常見的內(nèi)部排序算法有:插入排序、希爾排序、選擇排序、冒泡排序、歸并排序、快速排序、堆排序、基數(shù)排序等。用一張圖概括: showImg(https://segmentfault.com/img/bVNwuO?w=966&h=...
閱讀 1357·2021-09-22 15:09
閱讀 2678·2021-08-20 09:38
閱讀 2418·2021-08-03 14:03
閱讀 877·2019-08-30 15:55
閱讀 3384·2019-08-30 12:59
閱讀 3561·2019-08-26 13:48
閱讀 1899·2019-08-26 11:40
閱讀 681·2019-08-26 10:30