亚洲开心激情网_亚洲欧美另类综合偷拍 _国产性天天综合网_欧美日韩午夜在线

學(xué)了鏈表牛刀小試,三種做法都吃透就算是學(xué)會了:環(huán)球關(guān)注
時間:2023-02-11 13:56:27  來源:Coder梁  
1
聽新聞

關(guān)注、星標(biāo)下方公眾號,和你一起成長


(資料圖片)

作者 | 梁唐

出品 | 公眾號:Coder梁(ID:Coder_LT)

大家好,我是梁唐。

今天我們繼續(xù)來挑戰(zhàn)鏈表,來看一道LeetCode當(dāng)中的一道經(jīng)典問題——206.反轉(zhuǎn)鏈表。

這道題在很多公司的面試和筆試題中都有出現(xiàn),我就曾經(jīng)遇到過。絕對算是鏈表領(lǐng)域的經(jīng)典例題了,如果最近剛好要參加面試筆試的同學(xué),那么千萬不要錯過,說不定就遇上了。

我們先來看看題面。

206. 反轉(zhuǎn)鏈表

給你單鏈表的頭節(jié)點 head,請你反轉(zhuǎn)鏈表,并返回反轉(zhuǎn)后的鏈表。

分析

題面還是比較直接的,就是讓我們將一個給定的鏈表來翻轉(zhuǎn)。

最簡單最取巧的方法當(dāng)然是先遍歷一遍鏈表,將所有元素存進數(shù)組之后,再認(rèn)為構(gòu)造一個鏈表。這樣做當(dāng)然不能說不行,只不過在面試當(dāng)中通常是無法令面試官滿意的。

因為我們根本沒有利用好給定我們的鏈表,額外地消耗了內(nèi)存空間。所以如果在面試當(dāng)中遇到,面試官是不會只滿足于聽到這樣的回答的。那么,我們又該如何在不創(chuàng)建新鏈表的前提下完成翻轉(zhuǎn)呢?

對于這個問題,這題很好心地在進階里面給了我們提示,可以使用迭代或者遞歸的方法。

我個人感覺這兩種方法難度差不多,不過從理解難度上來說,遞歸的方法更簡單直觀一些。

遞歸法

為什么說遞歸的方法稍微更直觀呢?因為我們可以把遞歸函數(shù)本身當(dāng)成是一個能夠在更小范圍內(nèi)運作的黑盒,接著,我們要做的就是像是套娃一樣,讓它能夠在更大的范圍當(dāng)中實現(xiàn)同樣的功能。

比如在這題當(dāng)中,我們要使用遞歸來實現(xiàn)reverseList函數(shù)。我們先假設(shè),它能夠在比當(dāng)前更小的范圍內(nèi)運行。對于當(dāng)前輸入來說是head開頭的鏈表,那么head->next開頭的鏈表就可以看成是比當(dāng)前范圍更小的范圍。我們假設(shè)reverseList能夠?qū)?code>head->next開頭的鏈表翻轉(zhuǎn),我們要在此基礎(chǔ)上構(gòu)造出以head開頭翻轉(zhuǎn)的結(jié)果。

假設(shè)當(dāng)前的輸入是[1, 2, 3, 4, 5],當(dāng)前head指向1。head->next指向[2, 3, 4, 5]。調(diào)用reverseList之后可以得到[5, 4, 3, 2]。所以我們要做的把head放到遞歸結(jié)果的末尾。

所以我們要做的就很簡單,只有兩步。第一步遞歸調(diào)用reverseList,傳入head->next拿到結(jié)果。第二步,將head插入到遞歸返回的鏈表末尾。

由于在本題中鏈表都是通過頭節(jié)點表示的,所以我們要先遍歷一次到達鏈表的結(jié)尾。不要忘了處理一下邊界情況,即head為空或者是head->next為空的情況。

這些都想明白的話,代碼自然也就出來了:

classSolution{public:ListNode*reverseList(ListNode*head){//處理邊界if(head==nullptr||head->next==nullptr)returnhead;//遞歸調(diào)用autocur=reverseList(head->next);//遍歷到鏈表的最后一個節(jié)點autotmp=cur;while(tmp->next!=nullptr)tmp=tmp->next;//插入headtmp->next=head;head->next=nullptr;returncur;}};

改進

這里我們?yōu)榱饲蟪鲦湵碜詈笠粋€節(jié)點在遞歸當(dāng)中使用了循環(huán),通過遍歷的方式去找了最末的節(jié)點。

實際上我們大可以不必如此,我們直接讓遞歸函數(shù)也返回末尾的指針即可。但這樣的話,我們就修改了返回值的類型,所以就要單獨寫一個遞歸來實現(xiàn)了。整體的原理和剛才是一樣的,只不過我們稍作加工,讓遞歸能夠既返回頭節(jié)點也返回尾節(jié)點。我們就不用再去額外遍歷了。

下面這段代碼的核心邏輯和之前是一樣的,只是優(yōu)化了遞歸返回的部分。

classSolution{public:pairreverse(ListNode*head){//處理邊界if(head==nullptr||head->next==nullptr)returnmake_pair(head,head);//遞歸autotmp=reverse(head->next);//將head插入到末尾tmp.second->next=head;head->next=nullptr;//返回新結(jié)果returnmake_pair(tmp.first,head);}ListNode*reverseList(ListNode*head){returnreverse(head).first;}};

迭代

理解完了遞歸的做法之后,我們再來思考:如果不使用遞歸,那么這道題又該怎么解決呢?

其實并不難,只需要理解一點就可以搞定。那就是對于鏈表來說,我們可以在任何節(jié)點插入元素。既然如此,我們既可以每次插入在末尾,自然也可以插入在頭部。如果我們每次插入元素都在頭部的話,得到的鏈表中的元素順序剛好和之前相反。

所以我們只需要再創(chuàng)建一個鏈表,一邊遍歷,一邊將讀取到的元素插入在新鏈表的頭部,最后返回即可。

這里可以使用之前介紹的虛擬頭節(jié)點的技巧來簡化代碼:

classSolution{public:ListNode*reverseList(ListNode*head){if(head==nullptr)returnhead;autopt=head;//新鏈表的虛擬頭節(jié)點autoret=newListNode(0);while(pt!=nullptr){autocur=pt;pt=pt->next;//插入在ret指針之后cur->next=ret->next;ret->next=cur;}returnret->next;}};

這道題雖然難度不大,但是正解的兩種方法都需要對鏈表這個數(shù)據(jù)結(jié)構(gòu)本身的特點有比較深入的理解以及一定的代碼能力才能通過。除了這題之外,還有LeetCode19 刪除鏈表倒數(shù)第n個元素這題也非常不錯。我個人認(rèn)為不如這題經(jīng)典,所以就不過多贅述了,感興趣的同學(xué)可以自己去找來練習(xí)一下。

感謝大家的閱讀,如果喜歡的話,懇請幫忙轉(zhuǎn)發(fā)擴散。算法學(xué)習(xí)之旅,與你同行。

喜歡本文的話不要忘記三連~

關(guān)鍵詞: 遞歸函數(shù) 因為我們

主站蜘蛛池模板: 日本高清视频一区二区三区| 国产精品毛片一区视频| 91精品国产综合久久香蕉922| 91精品视频观看| 日本久久亚洲电影| 国产精品成人av在线| 久久久久久国产精品久久| 91免费看片网站| 激情五月开心婷婷| 日韩免费一区二区三区| 91久久国产综合久久91精品网站 | 丝袜美腿亚洲一区二区| 久久亚洲国产精品日日av夜夜| 国产不卡在线观看| 欧美日韩免费精品| 日本精品一区二区三区不卡无字幕| 午夜精品久久久久久久99热| 国产精品免费视频xxxx| 国语自产精品视频在线看| 久久久久久久久91| 久久青草精品视频免费观看| 日韩在线中文视频| 无码人妻精品一区二区蜜桃百度| 91久久夜色精品国产网站| 国产精品极品在线| 97久久精品午夜一区二区| 美女视频久久| 国产精品免费观看高清| 国产精品99久久久久久www| 国产精品久久久av久久久| 国产精品自产拍在线观看中文| 国产精品久久久| 国产传媒久久久| 欧美日韩电影在线观看| 国产在线一区二区三区四区| 久久超碰亚洲| 欧美国产综合视频| 国产在线一区二区三区四区| 国产欧美日韩91| 国产伦精品免费视频| 国产专区精品视频|