【LeetCode算法系列题解】第21~25题

2023-11-01

LeetCode 21. 合并两个有序链表(简单)

【题目描述】

将两个升序链表合并为一个新的升序链表并返回。新链表是通过拼接给定的两个链表的所有节点组成的。

【示例1】

在这里插入图片描述

输入:l1 = [1,2,4], l2 = [1,3,4]
输出:[1,1,2,3,4,4]

【示例2】

输入:l1 = [], l2 = []
输出:[]

【示例3】

输入:l1 = [], l2 = [0]
输出:[0]

【提示】

两个链表的节点数目范围是 [0, 50]
− 100 ≤ N o d e . v a l ≤ 100 -100\le Node.val\le 100 100Node.val100
l1l2 均按非递减顺序排列

【分析】


直接模拟即可,每次取两个链表中较小的结点,接到新链表的后面,如果其中一个链表空了,则直接将另一个链表接到新链表的后面。


【代码】

/**
 * 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:
    ListNode* mergeTwoLists(ListNode* list1, ListNode* list2) {
        auto dummy = new ListNode(-1), cur = dummy;  // 用auto才能并排写
        while (list1 && list2)
            if (list1->val < list2->val) cur = cur->next = list1, list1 = list1->next;
            else cur = cur->next = list2, list2 = list2->next;
        if (list1) cur->next = list1;
        else if (list2) cur->next = list2;
        return dummy->next;
    }
};

LeetCode 22. 括号生成(中等)

【题目描述】

数字 n 代表生成括号的对数,请你设计一个函数,用于能够生成所有可能的并且有效的括号组合。

【示例1】

输入:n = 3
输出:["((()))","(()())","(())()","()(())","()()()"]

【示例2】

输入:n = 1
输出:["()"]

【提示】

1 ≤ n ≤ 8 1\le n\le 8 1n8

【分析】


本题只有小括号,对于这类问题,判断一个括号序列是否合法有一些很重要的推论:

  • 任意前缀中,( 数量一定大于等于 ) 数量(最重要);
  • () 的数量相等。

我们可以使用 DFS 搜索方案,对于每个位置,只要当前 ( 的数量小于 n n n 就可以填入,而填入 ) 需要满足当前 ) 的数量小于 n n n 且小于 ( 的数量。


【代码】

class Solution {
public:
    vector<string> res;
    vector<string> generateParenthesis(int n) {
        dfs(n, 0, 0, "");
        return res;
    }

    void dfs(int n, int lc, int rc, string now)  // lc和rc分别表示左右括号的数量
    {
        if (lc == n && rc == n) { res.push_back(now); return; }
        if (lc < n) dfs(n, lc + 1, rc, now + '(');
        if (rc < n && rc < lc) dfs(n, lc, rc + 1, now + ')');
    }
};

LeetCode 23. 合并K个升序链表(困难)

【题目描述】

给你一个链表数组,每个链表都已经按升序排列。
请你将所有链表合并到一个升序链表中,返回合并后的链表。

【示例1】

输入:lists = [[1,4,5],[1,3,4],[2,6]]
输出:[1,1,2,3,4,4,5,6]
解释:链表数组如下:
[
  1->4->5,
  1->3->4,
  2->6
]
将它们合并到一个有序链表中得到。
1->1->2->3->4->4->5->6

【示例2】

输入:lists = []
输出:[]

【示例3】

输入:lists = [[]]
输出:[]

【提示】

k = = l i s t s . l e n g t h k == lists.length k==lists.length
0 ≤ k ≤ 1 0 4 0\le k\le 10^4 0k104
0 ≤ l i s t s [ i ] . l e n g t h ≤ 500 0\le lists[i].length\le 500 0lists[i].length500
− 1 0 4 ≤ l i s t s [ i ] [ j ] ≤ 1 0 4 -10^4\le lists[i][j]\le 10^4 104lists[i][j]104
lists[i]升序排列
lists[i].length 的总和不超过 1 0 4 10^4 104

【分析】


和第21题差不多,我们每次从这 K K K 个链表中找出最小的结点,将其接到新链表的后面,可以使用一个小根堆(优先队列 priority_queue 来维护最小值)。需要注意的是我们需要写一个排序算法,C++ 中优先队列的排序算法不是传入一个函数,而是重载了 () 的结构体,具体实现方式见代码部分。


【代码】

/**
 * 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:
    struct Cmp
    {
        // 默认是大根堆,因此用大于号翻转为小根堆,注意一定要有{}
        bool operator() (ListNode*& a, ListNode*& b) { return a->val > b->val; } 
    };

    ListNode* mergeKLists(vector<ListNode*>& lists) {
        auto dummy = new ListNode(-1), cur = dummy;
        priority_queue<ListNode*, vector<ListNode*>, Cmp> Q;  // 传入比较结构体
        for (auto list: lists)
            if (list) Q.push(list);  // 判断是否为空
        while (Q.size())
        {
            auto t = Q.top(); Q.pop();
            cur = cur->next = t;
            if (t->next) Q.push(t->next);
        }
        return dummy->next;
    }
};

LeetCode 24. 两两交换链表中的节点(中等)

【题目描述】

给你一个链表,两两交换其中相邻的节点,并返回交换后链表的头节点。你必须在不修改节点内部的值的情况下完成本题(即,只能进行节点交换)。

【示例1】

在这里插入图片描述

输入:head = [1,2,3,4]
输出:[2,1,4,3]

【示例2】

输入:head = [1,2,3,4]
输出:[2,1,4,3]

【示例3】

输入:head = [1]
输出:[1]

【提示】

链表中节点的数目在范围 [0, 100]
0 ≤ N o d e . v a l ≤ 100 0\le Node.val\le 100 0Node.val100

【分析】


我们通过画图可以更加直观地分析:

在这里插入图片描述

具体步骤如下:

  1. P->nextP->next->next 存在时,将其分别记为 AB
  2. P->next = B
  3. A->next = B->next
  4. B->next = A
  5. P = A

【代码】

/**
 * 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:
    ListNode* swapPairs(ListNode* head) {
        auto dummy = new ListNode(-1), cur = dummy;
        dummy->next = head;
        while (cur->next && cur->next->next)
        {
            auto a = cur->next, b = cur->next->next;
            cur->next = b, a->next = b->next, b->next = a, cur = a;
        }
        return dummy->next;
    }
};

LeetCode 25. K 个一组翻转链表(困难)

【题目描述】

给你链表的头节点 head,每 k 个节点一组进行翻转,请你返回修改后的链表。
k 是一个正整数,它的值小于或等于链表的长度。如果节点总数不是 k 的整数倍,那么请将最后剩余的节点保持原有顺序。
你不能只是单纯的改变节点内部的值,而是需要实际进行节点交换。

【示例1】

在这里插入图片描述

输入:head = [1,2,3,4,5], k = 2
输出:[2,1,4,3,5]

【示例2】

输入:head = [1,2,3,4,5], k = 3
输出:[3,2,1,4,5]

【提示】

链表中的节点数目为 n n n
1 ≤ k ≤ n ≤ 5000 1\le k\le n\le 5000 1kn5000
0 ≤ N o d e . v a l ≤ 1000 0\le Node.val\le 1000 0Node.val1000

【分析】


和上一题的分析类似,我们先画出示意图:

在这里插入图片描述

具体步骤如下:

  1. 先遍历一次看看 P 后面是否存在 K 个结点,若不存在直接返回结果,若存在则记最后一个结点为 V
  2. 分别将 P->nextP->next->next 记为 AB
  3. P->next = V
  4. A->next = V->next
  5. P = A
  6. B->next 记为 C
  7. B->next = A
  8. A = B, B = C
  9. 重复6~8共 K − 1 K-1 K1 次。

【代码】

/**
 * 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:
    ListNode* reverseKGroup(ListNode* head, int k) {
        auto dummy = new ListNode(-1), cur = dummy;
        dummy->next = head;
        while (true)
        {
            auto v = cur;
            for (int i = 0; i < k; i++)
                if (v->next == nullptr) return dummy->next;
                else v = v->next;
            auto a = cur->next, b = cur->next->next;
            cur->next = v, a->next = v->next, cur = a;
            for (int i = 0; i < k - 1; i++)
            {
                auto c = b->next;
                b->next = a, a = b, b = c;
            }
        }
        return dummy->next;  // 此行避免报错,不会执行
    }
};
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

【LeetCode算法系列题解】第21~25题 的相关文章

随机推荐

  • ceph中的Pools、PGs和OSDs介绍(tmp)

    2019独角兽企业重金招聘Python工程师标准 gt gt gt How are Placement Groups used A placement group PG aggregates objects within a pool be
  • Python-栈结构

    栈 stack 又名堆栈 它是一种运算受限的线性表 栈只能在一端进行插入和删除操作 它按照先进后出 FILO 的原则存储数据 先进入的数据被压入栈底 最后的数据在栈顶 栈也可以看成是 FILO 的队列 class Stack object
  • String类常用方法

    红色为常用的方法 1 和长度有关的方法 得到一个字符串的字符长度 String s abc s length 2 和数组有关的方法 返回类型 方法名 作用 byte getBytes 将一个字符串转换成字节数组 char toCharArr
  • mysql对表中列的操作_mysql对表基本操作

    一 对表的操作 1 添加新的字段 alter table 表名 add name varchar 20 2 删除表中已有的字段 alter table 表名 drop name 3 修改表中已有的字段 alter table 表名 chan
  • js 计算两个日期之间的相差的天数

    将两个日期都转换为毫秒相减后 将减下来的毫秒数转换为天数 就可以得到两个日期之间相差的天数了 接受的日期格式为 2023 1 31 2023 2 28 的日期字符串 const getDaysApart date val date vals
  • ubuntu下jmxtrans 安装

    JAVA 监控内容收集之 Jmxtrans 它是一个为应用程序 设备 系统等管理功能的框架 通常可以用来监控和管理Java应用系统 1 拷贝jmxtrans至LS1上 scp jmxtrans 251 deb LS1 2 安装jmxtran
  • Google Chrome在Windows7安装离线版

    前言 今天因为旧版chrome老是要报更新 所以安装了个新版 因为被墙原因 许多网友会遇到一些安装chrome的问题 所以今天分享一下安装教程 安装chrome 1 前往chrome官网 可以看到链接地址是http www google c
  • 如何构造测试数据

    前言 我这里只是专注于生成CSV等测试数据文件 每次构造测试数据的时候就很头疼 之前自己简单造个两三行还行 造多了就有些费脑细胞了 抽出些时间来专门找一下有没有相应工具 小数据量测试数据 小数据量测试数据使用在线的网站就行 10W以内的数据
  • 【Python】使用Python根据BV号爬取对应B站视频下的所有评论(包括评论下的回复)

    Python 使用Python根据BV号爬取对应B站视频下的所有评论 包括评论下的回复 本文写于2020 4 27 当你阅读到本文的时候如果因为下列原因导致本文代码无法正常工作 本人概不负责 B站的页面和API接口的变动 B站为页面和API
  • 操作系统笔记(手写)

    前言 这学期开始学习计算机网络 操作系统和Java程序设计 这些课的重要性不言而喻 对于我这种纯粹的小白来说 压力真得很大 自己水平有限 领悟能力较差 学习接受能力很慢 不知道怎样才能真真的学懂 学会这些东西 所以就先跟着学校安排的网课和配
  • 常见的数据结构与算法

    文章目录 前言 一 常见的数据结构 1 数组 2 链表 3 栈 4 队列 5 树 二 排序 1 基本的排序算法 2 常考的排序算法 3 其他排序算法 三 递归与回溯 1 递归 2 回溯 四 深度与广度优先搜索 1 深度优先搜索 2 广度优先
  • 伴随矩阵介绍及C++实现

    在线性代数中 一个方形矩阵的伴随矩阵是一个类似于逆矩阵的概念 如果矩阵可逆 那么它的逆矩阵和它的伴随矩阵之间只差一个系数 然而 伴随矩阵对不可逆的矩阵也有定义 并且不需要用到除法 设R是一个交换环 在抽象代数之分支环论中 一个交换环 com
  • 【vue】vue子孙组件传值(多级嵌套)attrs listeners

    如果vue开发遇到多层嵌套 子孙组件之间传值 可以使用 attrs listeners传值 示例如下 孙子组件
  • 装上这10个插件,PyCharm才是无敌的存在

    pycharm是一款强大的python集成开发环境 带有一整套python开发工具 今天就给大家介绍几款非常好用的插件 首先插件的下载方法 进入File gt Settings gt Plugins 根据需要搜索插件名称 记得是在Marke
  • db是哪个城市的缩写_全国所有城市拼音及缩写

    北京 BEIJING BJ 上海 SHANGHAI SH 天津 TIANJIN TJ 重庆 CHONGQING ZQ 阿克苏 AKESU AKS 安宁 ANNING AN 安庆 ANQING AQ 鞍山 ANSHAN AS 安顺 ANSHU
  • 分享一款开源堡垒机-jumpserver

    JumpServer是由FIT2CLOUD 飞致远 公司旗下一款开源的堡垒机 这款也是全球首款开源的堡垒机 使用 GNU GPL v2 0 开源协议 是符合 4A 规范的运维安全审计系统 使用 Python 开发 遵循 Web 2 0 规范
  • java basefont_itext 文本域 字体样式设置

    使用acroFields setFieldProperty nameField textfont baseFont null 的方式不能加粗 因为第三个参数必须是BaseFont类型 不能是Font类型 可以使用下面的方式加粗 BaseFo
  • 判断环形链表是否有环??返回环形链表的入口点!!

    上次笔者写了一篇大概有7个题的链表相关的题目 解析 感觉还不错 感兴趣的各位老铁 可以点一下链接进行欣赏 做几个与链表相关的题吧 https blog csdn net weixin 64308540 article details 128
  • 牧师与魔鬼 -- version2 动作分离

    目录 一 基本操作演练 1 下载 Fantasy Skybox FREE 构建自己的游戏场景 2 写一个简单的总结 总结游戏对象的使用 二 编程实践 1 牧师与魔鬼 动作分离版 面向对象的游戏编程 动作管理器的设计思想 动作管理器的设计类图
  • 【LeetCode算法系列题解】第21~25题

    CONTENTS LeetCode 21 合并两个有序链表 简单 LeetCode 22 括号生成 中等 LeetCode 23 合并K个升序链表 困难 LeetCode 24 两两交换链表中的节点 中等 LeetCode 25 K 个一组