鏈表問題(二)-LeetCode 147、876、234、817、92(鏈表的中點,快慢指針)

  • 2019 年 12 月 13 日
  • 筆記

1

編程題

【LeetCode #147】對鏈表進行插入排序

對鏈表進行插入排序。

示例 1: 輸入: 4->2->1->3 輸出: 1->2->3->4

解題思路:

首先判斷兩個相鄰節點的大小,如果head->val > head->next->val,則需要將head->next->val插入到從dummy節點開始遍歷第一個大於head->next->val節點的前面!好好理解這句話!在進行插入的時候,首先使用cur指針標記head->next節點,並改變head->next的指向。從而將待插入節點分離!接著就是普通的插入操作了!

/**   * Definition for singly-linked list.   * struct ListNode {   *     int val;   *     ListNode *next;   *     ListNode(int x) : val(x), next(NULL) {}   * };   */  class Solution {  public:      ListNode* insertionSortList(ListNode* head) {          if(head == nullptr || head->next == nullptr) return head;          ListNode* dummy = new ListNode(), *pre = nullptr;          dummy->next = head;          while(head->next != nullptr){              if(head->val <= head->next->val){  // 無需進行插入操作                  head = head->next;                  continue;              }              pre = dummy;              while(pre->next->val < head->next->val){                  pre = pre->next;              }              ListNode* cur = head->next;              head->next = cur->next;              cur->next = pre->next;              pre->next = cur;          }          return dummy->next;      }  };  

來源:力扣(LeetCode) 鏈接:https://leetcode-cn.com/problems/insertion-sort-list/

【LeetCode #876】鏈表的中間結點

給定一個帶有頭結點 head 的非空單鏈表,返回鏈表的中間結點。 如果有兩個中間結點,則返回第二個中間結點。

示例 1: 輸入:[1,2,3,4,5] 輸出:此列表中的結點 3 (序列化形式:[3,4,5]) 返回的結點值為 3 。(測評系統對該結點序列化表述是 [3,4,5])。 注意,我們返回了一個 ListNode 類型的對象 ans,這樣: ans.val = 3, ans.next.val = 4, ans.next.next.val = 5, 以及 ans.next.next.next = NULL.

解題思路:快慢指針,注意與下一題中的迴文鏈表的中間結點進行區別!

/**   * Definition for singly-linked list.   * struct ListNode {   *     int val;   *     ListNode *next;   *     ListNode(int x) : val(x), next(NULL) {}   * };   */  class Solution {  public:      ListNode* middleNode(ListNode* head) {          if(head == nullptr || head->next == nullptr) return head;          ListNode* slow = head, *fast = head;          while(fast != nullptr && fast->next != nullptr){              slow = slow->next;              fast = fast->next->next;          }          return slow;      }  };  

來源:力扣(LeetCode) 鏈接:https://leetcode-cn.com/problems/middle-of-the-linked-list

【LeetCode #234】迴文鏈表

請判斷一個鏈表是否為迴文鏈表。

示例 1: 輸入: 1->2 輸出: false

解題思路:

找到中點(奇數時不是真正的中點,注意與上題區別),然後反轉後面的鏈表,再進行節點值得比較即可!

/**   * Definition for singly-linked list.   * struct ListNode {   *     int val;   *     ListNode *next;   *     ListNode(int x) : val(x), next(NULL) {}   * };   */  class Solution {  public:      bool isPalindrome(ListNode* head) {          ListNode* slow = head, *fast = head, *pre = nullptr;          while(fast != nullptr){              slow = slow->next;              fast = fast->next ? fast->next->next : nullptr;          }   // 這裡的中點偶數時為第二個節點,奇數時為中點的下一個節點!          while(slow != nullptr){              ListNode* next = slow->next;              slow->next = pre;              pre = slow;              slow = next;          }          while(pre && head){              if(pre->val != head->val){                  return false;              }              pre = pre->next;              head = head->next;          }          return true;      }  };  

來源:力扣(LeetCode) 鏈接:https://leetcode-cn.com/problems/middle-of-the-linked-list

【LeetCode #817】鏈表組件

給定一個鏈表(鏈表結點包含一個整型值)的頭結點 head。 同時給定列表 G,該列表是上述鏈表中整型值的一個子集。 返回列表 G 中組件的個數,這裡對組件的定義為:鏈表中一段最長連續結點的值(該值必須在列表 G 中)構成的集合。

示例 1: 輸入: head: 0->1->2->3 G = [0, 1, 3] 輸出: 2 解釋: 鏈表中,0 和 1 是相連接的,且 G 中不包含 2,所以 [0, 1] 是 G 的一個組件,同理 [3] 也是一個組件,故返回 2。

解題思路:

一開始題目沒有看太懂,可能語文不太好!他的意思就是,如果連續相鄰的鏈表節點值都在G這個數組中,那麼這為一個組件,如果不在,則忽略!最終答案是G中有多少個組件。 比如0->1->3->2, 而G = [1,3,0] 則答案為 1。 我們查詢可以在set中進行,設置布爾變數flag的作用是:用true標記這是一個組件,而false表示組件斷開了,此時res++, 但有可能我們遍歷的節點一直都不在G中。因此res++需要一個條件限制,即flag = true.

/**   * Definition for singly-linked list.   * struct ListNode {   *     int val;   *     ListNode *next;   *     ListNode(int x) : val(x), next(NULL) {}   * };   */  class Solution {  public:      int numComponents(ListNode* head, vector<int>& G) {          set<int> s(G.begin(), G.end());          int res = ;          bool flag = false;   // 找到標記為true          while(head != nullptr){              auto it = s.find(head->val);              if(it != s.end()){   // 找到了                  flag = true;              }else{               // 沒找到,前面的作為一個組件                  if(flag){ res++; }                  flag = false;              }              head = head->next;          }          if(flag) res++;          return res;      }  };  

來源:力扣(LeetCode) 鏈接:https://leetcode-cn.com/problems/linked-list-components

【LeetCode #92】反轉鏈表 II

反轉從位置 m 到 n 的鏈表。請使用一趟掃描完成反轉。

說明: 1 ≤ m ≤ n ≤ 鏈表長度。

示例: 輸入: 1->2->3->4->5->NULL, m = 2, n = 4 輸出: 1->4->3->2->5->NULL

/**   * Definition for singly-linked list.   * struct ListNode {   *     int val;   *     ListNode *next;   *     ListNode(int x) : val(x), next(NULL) {}   * };   */  class Solution {  public:      ListNode* reverseBetween(ListNode* head, int m, int n) {          ListNode* dummy = new ListNode();          dummy->next = head;          ListNode* pre = dummy;          for(int i = ; i < m; i++){              pre = pre->next;          }          head = pre->next;          for(int i = m; i < n; i++){              ListNode* tmp = head->next;              head->next = tmp->next;              tmp->next = pre->next;              pre->next = tmp;          }          return dummy->next;      }  };  

來源:力扣(LeetCode) 鏈接:https://leetcode-cn.com/problems/reverse-linked-list-i