链表问题(一)-LeetCode 138、382、203、143、141、142(有环链表入环节点,复制复杂链表)

  • 2019 年 12 月 13 日
  • 筆記

链表问题(一):LeetCode #138 382 203 143 141 142

1

编程题

【LeetCode #138】复制带随机指针的复杂链表

解题思路: 如图所示:

class Solution {  public:      Node* copyRandomList(Node* head) {          if(head == nullptr)  return nullptr;          Node* cur = head;          Node* pNext = nullptr;          // 第一步:1 -> 1' -> 2 -> 2' ...          while(cur != nullptr){              pNext = cur->next;              cur->next = new Node(cur->val);              cur->next->next = pNext;              cur = pNext;          }          // 第二步:修改random指针为cur->random->next          cur = head;          Node* copyNode = nullptr;          while(cur != nullptr){              pNext = cur->next->next;              copyNode = cur->next;              copyNode->random = ((cur->random == nullptr) ? nullptr : cur->random->next);              cur = pNext;          }          cur = head;          Node* res = cur->next;          while(cur != nullptr){              pNext = cur->next->next;              copyNode = cur->next;              cur->next = pNext;              copyNode->next = (pNext != nullptr) ? pNext->next : nullptr;              cur = pNext;          }          return res;      }  };  

来源:力扣(LeetCode) 链接:https://leetcode-cn.com/problems/copy-list-with-random-pointer/

【LeetCode #382】链表的随机节点

给定一个单链表,随机选择链表的一个节点,并返回相应的节点值。保证每个节点被选的概率一样。 进阶: 如果链表十分大且长度未知,如何解决这个问题?你能否使用常数级空间复杂度实现?

示例: // 初始化一个单链表 [1,2,3]. ListNode head = new ListNode(1); head.next = new ListNode(2); head.next.next = new ListNode(3); Solution solution = new Solution(head); // getRandom()方法应随机返回1,2,3中的一个,保证每个元素被返回的概率相等。 solution.getRandom();

解题思路:

首先遍历得到链表的长度,然后使用rand函数获取随机值,在找到对应节点!

class Solution {  public:      /** @param head The linked list's head.          Note that the head is guaranteed to be not null, so it contains at least one node. */      Solution(ListNode* head) {          phead = head;          while(head != nullptr){              size++;              head = head->next;          }      }        /** Returns a random node's value. */      int getRandom() {          ListNode* cur = phead;          int k = rand() % size;          for(int i = ; i < k; i++){              cur = cur->next;          }          return cur->val;      }  private:      int size = ;      ListNode* phead;  };  

来源:力扣(LeetCode) 链接:https://leetcode-cn.com/problems/linked-list-random-node

【LeetCode #203】移除链表元素

删除链表中等于给定值 val 的所有节点。

示例: 输入: 1->2->6->3->4->5->6, val = 6 输出: 1->2->3->4->5

解题思路:

由于有可能是头结点被删除,因此需要使用哨兵节点dummy,在遍历删除中只使用一个指针pre,由于在循环中使用了pre->next->next, 因此while中的条件应该为pre->next != nullptr.

class Solution {  public:      ListNode* removeElements(ListNode* head, int val) {          ListNode* dummy = new ListNode(-1);  // pre指向dummy节点          dummy->next = head;          ListNode* pre = dummy;          while(pre->next != nullptr){              if(pre->next->val == val){                  pre->next = pre->next->next;                }else{                  pre = pre->next;              }          }          return dummy->next;      }  };  

来源:力扣(LeetCode) 链接:https://leetcode-cn.com/problems/remove-linked-list-elements/

【LeetCode #143】重排链表

给定一个单链表 L:L0→L1→…→Ln-1→Ln , 将其重新排列后变为:L0→Ln→L1→Ln-1→L2→Ln-2→… 你不能只是单纯的改变节点内部的值,而是需要实际的进行节点交换。

示例 1: 给定链表 1->2->3->4, 重新排列为 1->4->2->3.

解题思路:

将每个节点的指针放入vector中,然后进行连接,直接原地O(1)太难了,我记不住。。。

class Solution {  public:      void reorderList(ListNode* head) {          if(head == nullptr) return;          vector<ListNode*> vec;          while(head != nullptr){              vec.push_back(head);              head = head->next;          }          int left = , right = vec.size()-1;          while(left < right){              vec[left]->next = vec[right];              vec[right--]->next = vec[++left];          }          vec[left]->next = nullptr;      }  };  

来源:力扣(LeetCode) 链接:https://leetcode-cn.com/problems/reorder-list

【LeetCode #141】环形链表

给定一个链表,判断链表中是否有环。

为了表示给定链表中的环,我们使用整数 pos 来表示链表尾连接到链表中的位置(索引从 0 开始)。如果 pos 是 -1,则在该链表中没有环。

示例 1: 输入:head = [3,2,0,-4], pos = 1 输出:true 解释:链表中有一个环,其尾部连接到第二个节点。

解题思路:

使用快慢指针,慢指针一次走一步,快指针一次走两步,如果有环,必定相遇,否则快指针会指向nullptr。

class Solution {  public:      bool hasCycle(ListNode *head) {          if(head == nullptr ||          head->next == nullptr ||          head->next->next == nullptr) return false;   // 两个节点以上才会有环            ListNode *slow = head, *fast = head;          while(fast != nullptr){              slow = slow->next;              fast = (fast->next != nullptr) ? fast->next->next : nullptr;              if(slow == fast) return true;          }          return false;      }  };  

来源:力扣(LeetCode) 链接:https://leetcode-cn.com/problems/linked-list-cycle

【LeetCode #142】环形链表II

给定一个链表,返回链表开始入环的第一个节点。如果链表无环,则返回 null。 为了表示给定链表中的环,我们使用整数 pos 来表示链表尾连接到链表中的位置(索引从 0 开始)。如果 pos 是 -1,则在该链表中没有环。

说明:不允许修改给定的链表。

示例 1: 输入:head = [3,2,0,-4], pos = 1 输出:tail connects to node index 1 解释:链表中有一个环,其尾部连接到第二个节点。

解题思路:

仍然是快慢指针,快指针一次走两步,慢指针一次走一步,上一题只需要判断有环无环即可,因此只需要相遇一次就好了,但这一题需要返回入环节点,因此需要相遇两次!在第一次相遇后,将慢指针指向头部位置,同时快指针改为一次走一步,第二次相遇的节点即为入环节点!

class Solution {  public:      ListNode *detectCycle(ListNode *head) {          if(head == nullptr || head->next == nullptr ||          head->next->next == nullptr)  return nullptr;            ListNode *slow = head, *fast = head;          while(fast != nullptr){              slow = slow->next;              fast = (fast->next != nullptr) ? fast->next->next : nullptr;              if(slow == fast){                  break;              }          }          if(fast == nullptr)  return nullptr;          slow = head;          while(slow != fast){              slow = slow->next;              fast = fast->next;          }          return slow;      }  };  

来源:力扣(LeetCode) 链接:https://leetcode-cn.com/problems/linked-list-cycle-ii