剑指 Offer II 026. 重排链表 mid
给定一个单链表 L
的头节点 head
,单链表 L
表示为:
L0 → L1 → … → Ln-1 → Ln
请将其重新排列后变为:
L0 → Ln → L1 → Ln-1 → L2 → Ln-2 → …
不能只是单纯的改变节点内部的值,而是需要实际的进行节点交换。
输入: head = [1,2,3,4]
输出: [1,4,2,3]
输入: head = [1,2,3,4,5]
输出: [1,5,2,4,3]
分析:栈
我们首先用 快慢指针 找到链表的中点,将中点后面一段的结点 塞到一个栈stack
里(这样就可以把结点的顺序反过来,
栈顶元素就是最后一个结点),接着再将栈中的结点弹出,插入到第一段中的位置(每两个节点中间插入一个)。
当链表结点数量是 奇数 时,slow
指针需要往后移动一次才行。
当链表结点数量是 偶数 时。
时间复杂度:O(n)O(n)O(n)
代码:
/*** Definition for singly-linked list.* struct ListNode {* int val;* ListNode *next;* ListNode() : val(0), next(nullptr) {}* ListNode(int x) : val(x), next(nullptr) {}* ListNode(int x, ListNode *next) : val(x), next(next) {}* };*/
class Solution {
public:void reorderList(ListNode* head) {ListNode * fast = head , *slow = head , *pre = nullptr;//快慢指针 求链表的中点while(fast && fast->next){pre = slow;fast = fast->next->next;slow = slow->next;}//根据结点数量是 奇数 还是 偶数分别处理,将链表分为两段if(fast){pre = slow;slow = slow->next;pre->next = nullptr;}else{pre->next = nullptr;}//用栈存储第二段的结点stack stk;while(slow){stk.push(slow->val);slow = slow->next;}//拼接到一起ListNode * cur = head;while(!stk.empty()){ListNode *nextNode = cur->next;int val = stk.top();stk.pop();ListNode *node = new ListNode(val);cur->next = node;node->next = nextNode;cur = nextNode;}}
};