判斷單連結串列是否存在環,判斷兩個連結串列是否相交問題詳解
有一個單連結串列,其中可能有一個環,也就是某個節點的next指向的是連結串列中在它之前的節點,這樣在連結串列的尾部形成一環。
問題:
1、如何判斷一個連結串列是不是這類連結串列?
2、如果連結串列為存在環,如何找到環的入口點?
解答:
一、判斷連結串列是否存在環,辦法為:
設定兩個指標(fast, slow),初始值都指向頭,slow每次前進一步,fast每次前進二步,如果連結串列存在環,則fast必定先進入環,而slow後進入環,兩個指標必定相遇。(當然,fast先行頭到尾部為NULL,則為無環連結串列)程式如下:
二、找到環的入口點
當fast若與slow相遇時,slow肯定沒有走遍歷完連結串列,而fast已經在環內迴圈了n圈(1<=n)。假設slow走了s步,則fast走了2s步(fast步數還等於s 加上在環上多轉的n圈),設環長為r,則:
2s = s + nr
s= nr
設整個連結串列長L,入口環與相遇點距離為x,起點到環入口點的距離為a。
a + x = nr
a + x = (n – 1)r +r = (n-1)r + L - a
a = (n-1)r + (L – a – x)
(L – a – x)為相遇點到環入口點的距離,由此可知,從連結串列頭到環入口點等於(n-1)迴圈內環+相遇點到環入口點,於是我們從連結串列頭、與相遇點分別設一個指標,每次各走一步,兩個指標必定相遇,且相遇第一點為環入口點。程式描述如下:
擴充套件問題:
判斷兩個單連結串列是否相交,如果相交,給出相交的第一個點(兩個連結串列都不存在環)。
比較好的方法有兩個:
一、將其中一個連結串列首尾相連,檢測另外一個連結串列是否存在環,如果存在,則兩個連結串列相交,而檢測出來的依賴環入口即為相交的第一個點。
二、如果兩個連結串列相交,那個兩個連結串列從相交點到連結串列結束都是相同的節點,我們可以先遍歷一個連結串列,直到尾部,再遍歷另外一個連結串列,如果也可以走到同樣的結尾點,則兩個連結串列相交。
這時我們記下兩個連結串列length,再遍歷一次,長連結串列節點先出發前進(lengthMax-lengthMin)步,之後兩個連結串列同時前進,每次一步,相遇的第一點即為兩個連結串列相交的第一個點。
問題:
1、如何判斷一個連結串列是不是這類連結串列?
2、如果連結串列為存在環,如何找到環的入口點?
解答:
一、判斷連結串列是否存在環,辦法為:
設定兩個指標(fast, slow),初始值都指向頭,slow每次前進一步,fast每次前進二步,如果連結串列存在環,則fast必定先進入環,而slow後進入環,兩個指標必定相遇。(當然,fast先行頭到尾部為NULL,則為無環連結串列)程式如下:
bool IsExitsLoop(slist *head)
{
slist *slow = head, *fast = head;
while ( fast && fast->next )
{
slow = slow->next;
fast = fast->next->next;
if ( slow == fast ) break;
}
return !(fast == NULL || fast->next == NULL);
}
{
slist *slow = head, *fast = head;
while ( fast && fast->next )
{
slow = slow->next;
fast = fast->next->next;
if ( slow == fast ) break;
}
return !(fast == NULL || fast->next == NULL);
}
二、找到環的入口點
當fast若與slow相遇時,slow肯定沒有走遍歷完連結串列,而fast已經在環內迴圈了n圈(1<=n)。假設slow走了s步,則fast走了2s步(fast步數還等於s 加上在環上多轉的n圈),設環長為r,則:
2s = s + nr
s= nr
設整個連結串列長L,入口環與相遇點距離為x,起點到環入口點的距離為a。
a + x = nr
a + x = (n – 1)r +r = (n-1)r + L - a
a = (n-1)r + (L – a – x)
(L – a – x)為相遇點到環入口點的距離,由此可知,從連結串列頭到環入口點等於(n-1)迴圈內環+相遇點到環入口點,於是我們從連結串列頭、與相遇點分別設一個指標,每次各走一步,兩個指標必定相遇,且相遇第一點為環入口點。程式描述如下:
slist* FindLoopPort(slist *head)
{
slist *slow = head, *fast = head;
while ( fast && fast->next )
{
slow = slow->next;
fast = fast->next->next;
if ( slow == fast ) break;
}
if (fast == NULL || fast->next == NULL)
return NULL;
slow = head;
while (slow != fast)
{
slow = slow->next;
fast = fast->next;
}
return slow;
}
{
slist *slow = head, *fast = head;
while ( fast && fast->next )
{
slow = slow->next;
fast = fast->next->next;
if ( slow == fast ) break;
}
if (fast == NULL || fast->next == NULL)
return NULL;
slow = head;
while (slow != fast)
{
slow = slow->next;
fast = fast->next;
}
return slow;
}
擴充套件問題:
判斷兩個單連結串列是否相交,如果相交,給出相交的第一個點(兩個連結串列都不存在環)。
比較好的方法有兩個:
一、將其中一個連結串列首尾相連,檢測另外一個連結串列是否存在環,如果存在,則兩個連結串列相交,而檢測出來的依賴環入口即為相交的第一個點。
二、如果兩個連結串列相交,那個兩個連結串列從相交點到連結串列結束都是相同的節點,我們可以先遍歷一個連結串列,直到尾部,再遍歷另外一個連結串列,如果也可以走到同樣的結尾點,則兩個連結串列相交。
這時我們記下兩個連結串列length,再遍歷一次,長連結串列節點先出發前進(lengthMax-lengthMin)步,之後兩個連結串列同時前進,每次一步,相遇的第一點即為兩個連結串列相交的第一個點。
相關文章
- 連結串列面試題(十二)---判斷兩個都不帶環的連結串列是否相交面試題
- 連結串列面試題(九)---判斷一個連結串列是否帶環面試題
- 演算法題:反轉一個單連結串列&判斷連結串列是否有環演算法
- [2014校招筆試]判斷單連結串列是否有環?筆試
- Node判斷檔案是否連結
- 如何判斷一個連結地址是否有效
- 連結串列面試題(十三)---求兩個都不帶環的連結串列相交的結點面試題
- 漫畫演算法:如何判斷連結串列有環?演算法
- 牛客網高頻演算法題系列-BM6-判斷連結串列中是否有環演算法
- 判斷網路是否連線
- 41:判斷元素是否存在
- 牛客網高頻演算法題系列-BM13-判斷一個連結串列是否為迴文結構演算法
- 環形連結串列_相交連結串列_多數元素(java語言)Java
- Activiti判斷流程是否結束
- js判斷兩個物件是否相等JS物件
- JavaScript 判斷函式是否存在JavaScript函式
- postgresql如何判斷表是否存在SQL
- golang判斷檔案是否存在Golang
- vc判斷檔案是否存在
- jQuery如何判斷元素是否存在jQuery
- iOS判斷是否存在網路iOS
- 如何判斷Javascript物件是否存在JavaScript物件
- jQuery如何判斷一個元素是否存在jQuery
- 如何判斷一個jquery物件是否存在jQuery物件
- javascript判斷一個變數是否存在JavaScript變數
- js如何判斷一個物件是否存在JS物件
- java判斷百度雲分享連結是否失效Java
- 判斷url連結地址是否合法的例項程式碼
- 【圖解連結串列類面試題】環形連結串列圖解面試題
- 資料結構與演算法——連結串列 Linked List(單連結串列、雙向連結串列、單向環形連結串列-Josephu 問題)資料結構演算法
- 連結串列面試題(七)---合併兩個有序連結串列面試題
- 判斷元素是否在視口和元素相交
- (連結串列)連結串列的排序問題排序
- golang中判斷兩個slice是否相等與判斷值下的 陣列是否相等Golang陣列
- JavaScript判斷兩個變數是否相等JavaScript變數
- js 判斷兩個變數是否相等JS變數
- Linux Shell 判斷環境變數 是否存在Linux變數
- mysql如何判斷是否存在某個欄位MySql