LeetCode [24][25] k个一组反转链表

24. 两两交换链表中的节点 - 力扣(LeetCode)

25. K 个一组翻转链表 - 力扣(LeetCode)

扩展题目

  • 24解法1:
cpp 复制代码
// 迭代法难搞
    ListNode* swapPairs(ListNode* head) {
        if(head == nullptr || head->next == nullptr) return head;
        ListNode* ret = head->next;//用于返回的节点
        ListNode* p1 = head;
        ListNode* p2 = p1->next;
        ListNode* p3 = p2->next;//首先记录三个节点
        ListNode* tail = new ListNode();//记录本次操作的尾结点
        while(p3) 
        {
            p2->next = p1;//p2指向p1
            p1->next = p3;//p1指向下一次要翻转的p3,保证链表不断
            tail->next = p2;//上次的尾节点指向本次的头
            if(p3->next == nullptr) break;//如果此时p3->next为空,说明到链表尾部了,结束
            tail = p1;//更新tail,p1, p2, p3
            p1 = tail->next;
            p2 = p1->next;
            p3 = p2->next;
        }
        tail->next = p2;//结束循环后,tail指向当前的p2
        p2->next = p1;//p2指向p1
        p1->next = p3;//p1指向p3

        return ret;
    }
  • 24解法2
cpp 复制代码
//递归np
ListNode* swapPairs(ListNode* head) {
        if(head == nullptr || head->next == nullptr)
            return head;
        ListNode* p1 = head;
        ListNode* p2 = head->next;
        
        p1->next = swapPairs(p2->next);
        p2->next = p1;
        return p2;
    }
  • 25解法1
cpp 复制代码
// 迭代解法
ListNode* reverse(ListNode* head)
    {
        if(head == nullptr || head->next == nullptr) return head;
        ListNode* node1 = head;
        ListNode* node2 = node1->next;
        while(node1 && node2 && node2->next) 
        {
            node1 = node2;
            node2 = node2->next;
        }
        node1->next = nullptr;//node1为倒数第二个节点,node2为尾结点
        node2->next = reverse(head);//断开node1,node2指向翻转号的剩余head
        return node2;
    }
    ListNode* reverseKGroup(ListNode* head, int k) {
        if(head == nullptr || head->next == nullptr) return head;
        
        ListNode* tail = head;//记录当前中移动节点,可以放在while中
        ListNode* nextHead = head;//记录下一次翻转的头节点
        ListNode* res = nullptr;//用于返回
        ListNode* lastTail = nullptr;//记录上一次翻转的尾结点
        
        while(nextHead)
        {
            ListNode* curHead = nextHead;//此次的头结点
            tail = nextHead;//滑动节点
            int i = k;//单独一个变量记录i
            while(--i && tail) tail = tail->next;//找到本次翻转的尾结点,或者链表尾
            if(tail == nullptr)//如果已经到链表尾结点,不用再翻转
            {
                lastTail->next = nextHead;//上次的尾结点指向当前的尾,然后结束
                break;
            }
            nextHead = tail->next;//首先更新下一个Head
            tail->next = nullptr;//当前tail断开
            ListNode* ret = reverse(curHead);//把本次处理的链表翻转
            if(res == nullptr) res = ret;//用于返回res等于第一次翻转的链表
            if(lastTail)//使用每一个node的时,都要显式的判断一下是否为空
                lastTail->next = ret;//上次的tail指向本次翻转链表的头
            ListNode* curTail = ret;//求取本次翻转链表之后的tail
            while(curTail && curTail->next) curTail = curTail->next;
            lastTail = curTail;//更新此次tail
        }
        return res;
    }
  • 25解法2
cpp 复制代码
ListNode* reverse(ListNode* node1, ListNode* node2)
    {
        ListNode* pre = nullptr;
        ListNode* cur = node1;
        ListNode* next = nullptr;
        while(cur != node2)
        {
            next = cur->next;
            cur->next = pre;
            pre = cur;
            cur = next;
        }
        return pre;
    }
    ListNode* reverseKGroup(ListNode* head, int k) {
        if(head == nullptr || head ->next == nullptr) return head;
        ListNode* node1 = head;
        ListNode* node2 = head;          
        int i = k;
        while(i-- && node2) node2 = node2->next;//找到翻转链表的下一个结点
        if(node2 == nullptr && i >= 0)//如果到达链表尾部,直接返回
            return node1;
        ListNode* newHead = reverse(node1, node2);//翻转链表
        node1->next = reverseKGroup(node2, k);//递归进入下一轮
        return newHead;
    }
相关推荐
白羊by7 小时前
YOLOv1~v11 全版本核心演进总览
深度学习·算法·yolo
墨尘笔尖9 小时前
最大最小值降采样算法的优化
c++·算法
white-persist11 小时前
【vulhub shiro 漏洞复现】vulhub shiro CVE-2016-4437 Shiro反序列化漏洞复现详细分析解释
运维·服务器·网络·python·算法·安全·web安全
FL162386312912 小时前
基于C#winform部署软前景分割DAViD算法的onnx模型实现前景分割
开发语言·算法·c#
baizhigangqw12 小时前
启发式算法WebApp实验室:从搜索策略到群体智能的能力进阶
算法·启发式算法·web app
C雨后彩虹12 小时前
最多等和不相交连续子序列
java·数据结构·算法·华为·面试
cpp_250113 小时前
P2347 [NOIP 1996 提高组] 砝码称重
数据结构·c++·算法·题解·洛谷·noip·背包dp
Hugh-Yu-13012313 小时前
二元一次方程组求解器c++代码
开发语言·c++·算法
编程大师哥14 小时前
C++类和对象
开发语言·c++·算法
加农炮手Jinx14 小时前
LeetCode 146. LRU Cache 题解
算法·leetcode·力扣