給定單鏈表船侧,判斷是否有環(huán)欠气,如果有返回環(huán)入口

這是一題常見的面試題,考察求職者對鏈表的理解勺爱,題目在leetcode上:
Given a linked list, return the node where the cycle begins. If there is no cycle, return null.
首先來判斷鏈表有沒有環(huán)晃琳。看一個(gè)直觀的例子:假設(shè)小明和專業(yè)運(yùn)動(dòng)員在400米標(biāo)準(zhǔn)環(huán)形操場上比賽跑步琐鲁,由于長久缺乏鍛煉卫旱,小明勻速跑一圈需要2分鐘,而運(yùn)動(dòng)員勻速跑一圈只要1分鐘围段。那么如果兩個(gè)人同時(shí)顾翼、同一點(diǎn)出發(fā),1分鐘后小明跑了半圈奈泪,而運(yùn)動(dòng)員已經(jīng)跑完一圈适贸,開始第二圈灸芳;2分鐘時(shí),小明終于跑完一圈拜姿,而運(yùn)動(dòng)員剛好跑完第二圈烙样,整整比小明多跑一圈,兩人在終點(diǎn)相遇蕊肥。如果運(yùn)動(dòng)員在小明前方S米處和小明同時(shí)出發(fā)谒获,那么運(yùn)動(dòng)員會(huì)在小明跑完一圈前就追上小明(可憐的小明)。

回歸問題壁却,如果我們設(shè)定兩個(gè)指針p1批狱、p2,開始時(shí)指在鏈表頭head展东,p1代表小明赔硫,每次前進(jìn)一步;p2代表運(yùn)動(dòng)員盐肃,每次前進(jìn)兩步爪膊。如下圖環(huán)形鏈表中,6步之后p2追上p1恼蓬,再次相遇惊完。

環(huán)鏈表

對于一般鏈表,p1处硬、p2仍然同時(shí)從表頭A點(diǎn)出發(fā)小槐,由于前面有段非環(huán)的直線路程,當(dāng)p1到B點(diǎn)進(jìn)入環(huán)的時(shí)刻荷辕,p2已經(jīng)在環(huán)內(nèi)跑了一段路了凿跳,此刻開始,類似運(yùn)動(dòng)員和小明同時(shí)出發(fā)疮方,而運(yùn)動(dòng)員的出發(fā)點(diǎn)領(lǐng)先于小明控嗜,在不超過一圈的路程內(nèi),運(yùn)動(dòng)員能夠追上小明骡显,p2也能在一圈以內(nèi)趕上p1疆栏,假設(shè)兩指針在C點(diǎn)相遇,此時(shí)惫谤,p1==p2壁顶,利用這個(gè)條件,我們可以斷定鏈表有環(huán)溜歪。否則若专,當(dāng)運(yùn)動(dòng)員跑到了盡頭,還沒有和小明相遇蝴猪,說明他們的跑道沒有環(huán)调衰。

一般鏈表

再來看看環(huán)的入口在哪里膊爪。我們利用p1、p2在C點(diǎn)相遇的時(shí)刻來分析嚎莉,假設(shè)AB的距離為s米酬,BC的距離為r,CB的距離為l趋箩,p1走過距離為a = s + r則可以有以下關(guān)系:

(1)總長度 = s + r + l;
(2)環(huán)的長度 = r + l;
(3)p2走過距離 = 2a;
(4)p2 - p1 = a = s + r;
相遇時(shí)淮逻,p2比p1在環(huán)內(nèi)多走了n圈(n>=1): 
(5) p2 - p1 = n(r + l);
結(jié)合(4)阁簸、(5): n(r + l) = s + r;
nr - r = s - nl;
(n - 1)r = s - (n - 1)l -l;
(6) (n -1)(r + l) = s - l;

從(6)我們可以看出一些規(guī)律(注意:環(huán)的長度 = r + l),當(dāng)n = 1時(shí)哼丈,說明p2比p1多跑了一圈启妹,此時(shí),s = l醉旦。當(dāng)n > 1時(shí)饶米,說明p2比p1多跑了整整(n -1)圈再相遇,考慮s - l為環(huán)長度的(n -1)倍车胡,(n -1)(r + l) + l = s檬输,說明如果一個(gè)節(jié)點(diǎn)從A點(diǎn)開始,走s的距離和從C點(diǎn)開始走(n -1)圈再多l的距離剛好又能相遇在B點(diǎn)匈棘,B也正是環(huán)的入口丧慈。
分析完畢,再上C++代碼:

 struct ListNode {
      int val;
      ListNode *next;
      ListNode(int x) : val(x), next(NULL) {}
 };
ListNode *detectCycle(ListNode *head) {
        ListNode *p1 = head, *p2 = head;
        bool hasCycle = false;
        // 特殊情況判斷
        if(head == NULL)
            return NULL;
        // p2走得快主卫,當(dāng)p2->next == NULL時(shí)逃默,說明沒有環(huán)
        while(p2 && p2->next)
        {
            p1 = p1->next;
            p2 = p2->next->next;
            if(p1 == p2) 
            {
               //  有環(huán),立下flag
                hasCycle = true;
                break;
            }
        }
        if(!hasCycle)
        {
            return NULL;
        }
       // 一個(gè)從A前進(jìn)簇搅,一個(gè)從C前進(jìn)
        p2 = head;
        while(p1 != p2)
        {
            p1 = p1->next;
            p2 = p2->next;
        }
        return p1;
    }
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末完域,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子瘩将,更是在濱河造成了極大的恐慌吟税,老刑警劉巖,帶你破解...
    沈念sama閱讀 219,188評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件姿现,死亡現(xiàn)場離奇詭異肠仪,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)建钥,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,464評論 3 395
  • 文/潘曉璐 我一進(jìn)店門藤韵,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人熊经,你說我怎么就攤上這事泽艘∮眨” “怎么了?”我有些...
    開封第一講書人閱讀 165,562評論 0 356
  • 文/不壞的土叔 我叫張陵匹涮,是天一觀的道長天试。 經(jīng)常有香客問我,道長然低,這世上最難降的妖魔是什么喜每? 我笑而不...
    開封第一講書人閱讀 58,893評論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮雳攘,結(jié)果婚禮上带兜,老公的妹妹穿的比我還像新娘。我一直安慰自己吨灭,他們只是感情好刚照,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,917評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著喧兄,像睡著了一般无畔。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上吠冤,一...
    開封第一講書人閱讀 51,708評論 1 305
  • 那天浑彰,我揣著相機(jī)與錄音,去河邊找鬼拯辙。 笑死郭变,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的薄风。 我是一名探鬼主播饵较,決...
    沈念sama閱讀 40,430評論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼遭赂!你這毒婦竟也來了循诉?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,342評論 0 276
  • 序言:老撾萬榮一對情侶失蹤撇他,失蹤者是張志新(化名)和其女友劉穎茄猫,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體困肩,經(jīng)...
    沈念sama閱讀 45,801評論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡划纽,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,976評論 3 337
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了锌畸。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片勇劣。...
    茶點(diǎn)故事閱讀 40,115評論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出比默,到底是詐尸還是另有隱情幻捏,我是刑警寧澤,帶...
    沈念sama閱讀 35,804評論 5 346
  • 正文 年R本政府宣布命咐,位于F島的核電站篡九,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏醋奠。R本人自食惡果不足惜榛臼,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,458評論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望窜司。 院中可真熱鬧沛善,春花似錦、人聲如沸塞祈。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,008評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽织咧。三九已至,卻和暖如春漠秋,著一層夾襖步出監(jiān)牢的瞬間笙蒙,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,135評論 1 272
  • 我被黑心中介騙來泰國打工庆锦, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留捅位,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,365評論 3 373
  • 正文 我出身青樓搂抒,卻偏偏與公主長得像艇搀,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子求晶,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,055評論 2 355

推薦閱讀更多精彩內(nèi)容