链表中的节点每k个一组翻转

简介: 链表中的节点每k个一组翻转

描述

将给出的链表中的节点每 k 个一组翻转,返回翻转后的链表

如果链表中的节点数不是 k 的倍数,将最后剩下的节点保持原样

你不能更改节点中的值,只能更改节点本身。

数据范围: \ 0 \le n \le 2000 0≤n≤2000 , 1 \le k \le 20001≤k≤2000 ,链表中每个元素都满足 0 \le val \le 10000≤val≤1000

要求空间复杂度 O(1)O(1),时间复杂度 O(n)O(n)

例如:

给定的链表是 1\to2\to3\to4\to51→2→3→4→5

对于 k = 2k=2 , 你应该返回 2\to 1\to 4\to 3\to 52→1→4→3→5

对于 k = 3k=3 , 你应该返回 3\to2 \to1 \to 4\to 53→2→1→4→5


本题已AC


解题思路:

       拿到这题,一个想法就是用栈来写。至于为什么我会想到用栈呢?因为需要反向嘛,这里我看题目的要求,空间复杂度为 O(1)。第一个想法是保存值,通过 new一个新的节点了实现。但是仔细一想,好像不可以的,因为这样空间复杂度肯定不符合,突然灵机一动,我们可以用栈保存节点是指针,nice.

       1→23→4→5   k = 2时

      1→2→3→4→5  k = 3时  

       4和5是不用翻转的,所以如果用栈又会带来一个问题,因为栈结构是先进后出的,如果用栈来保存肯定会导致,4,5也会被翻转。

       这里有两个解决方法,

       1.把4,5再次入栈(入另一个栈),然后再输出 ok

       2.不妨我们不使用栈结构,我们使用双向队列(而且双向队列在前后两端进行插入删除的效率也挺高的,所以我们使用这个结构最佳了)

       我们哪一个例子来看:

       1→2→3→4→5  k = 3时  

       准备工作,一个指向该链表的指针 head。一个deque,一个中间变量 tmpk

       一些方便操作的辅助指针 s和ans

     

 

 

 

这里我们做一个循环,while(head != nullptr)     是的,一次遍历就可以搞定

此时退出循环,okk

判断 que是否为空或者tmpk是否等于 0

如果que不为空的话,说明que此时的节点是不需要翻转的

所以

那么这题就ok了,当然题目有一些细节,这个只能根据实际的测试清空来修改了

比如  {},2     {1,2},3    {1},2等

附上我通过的代码:

class Solution {
public:
    /**
     * 
     * @param head ListNode类 
     * @param k int整型 
     * @return ListNode类
     */
    ListNode* reverseKGroup(ListNode* head, int k) {
        // write code here
        deque<ListNode*> sta;
        int tmpk = k;
        ListNode *s = nullptr;
        ListNode *ans = nullptr;
        while(head != nullptr)
        {
            sta.push_back(head);
            head = head->next;
            tmpk--;
            if(tmpk == 0)
            {
                if(s == nullptr)
                {
                    s = sta.back();
                    sta.pop_back();
                    ans = s;
                }
                while(!sta.empty())
                {
                    s->next = sta.back();
                    sta.pop_back();
                    s = s->next;
                }
                tmpk = k;
            }
        }
        while(!sta.empty())
        {
            if(s == nullptr)
            {
                    s = sta.front();
                    sta.pop_front();
                    ans = s;
            }
            else
            {
                    s->next = sta.front();
                    sta.pop_front();
                s= s->next;
            }
        }
        if(s != nullptr)
        s->next = nullptr;
        return ans;
    }
};
相关文章
|
5月前
|
Python
【Leetcode刷题Python】25.K 个一组翻转链表
解决LeetCode "K 个一组翻转链表" 问题的三种方法:使用栈、尾插法和虚拟节点顺序法,并提供了每种方法的Python实现代码。
41 0
|
3月前
LeetCode第二十四题(两两交换链表中的节点)
这篇文章介绍了LeetCode第24题的解法,即如何通过使用三个指针(preNode, curNode, curNextNode)来两两交换链表中的节点,并提供了详细的代码实现。
34 0
LeetCode第二十四题(两两交换链表中的节点)
|
3月前
Leetcode第十九题(删除链表的倒数第N个节点)
LeetCode第19题要求删除链表的倒数第N个节点,可以通过快慢指针法在一次遍历中实现。
50 0
Leetcode第十九题(删除链表的倒数第N个节点)
05_删除链表的倒数第N个节点
05_删除链表的倒数第N个节点
|
3月前
(剑指offer)18、删除链表的节点—22、链表中倒数第K个节点—25、合并两个排序的链表—52、两个链表的第一个公共节点(2021.12.07)
(剑指offer)18、删除链表的节点—22、链表中倒数第K个节点—25、合并两个排序的链表—52、两个链表的第一个公共节点(2021.12.07)
59 0
|
5月前
|
算法
LeetCode第24题两两交换链表中的节点
这篇文章介绍了LeetCode第24题"两两交换链表中的节点"的解题方法,通过使用虚拟节点和前驱节点技巧,实现了链表中相邻节点的交换。
LeetCode第24题两两交换链表中的节点
04_两两交换链表中的节点
04_两两交换链表中的节点
|
5月前
|
Python
【Leetcode刷题Python】剑指 Offer 22. 链表中倒数第k个节点
Leetcode题目"剑指 Offer 22. 链表中倒数第k个节点"的Python解决方案,使用双指针法找到并返回链表中倒数第k个节点。
62 5
|
5月前
|
Python
【Leetcode刷题Python】剑指 Offer 18. 删除链表的节点
Leetcode题目"剑指 Offer 18. 删除链表的节点"的Python解决方案,通过使用双指针法找到并删除链表中值为特定数值的节点,然后返回更新后的链表头节点。
49 4
|
6月前
|
安全 云计算
云计算自旋锁问题之在线程安全地删除链表节点时,需要频繁加锁会影响性能如何解决
云计算自旋锁问题之在线程安全地删除链表节点时,需要频繁加锁会影响性能如何解决
62 2