链表问题(二)-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