当前位置: 首页 > news >正文

【题解】—— LeetCode一周小结44

🌟欢迎来到 我的博客 —— 探索技术的无限可能!


🌟博客的简介(文章目录)


【题解】—— 每日一道题目栏


上接:【题解】—— LeetCode一周小结43

28.冗余连接 II

题目链接:685. 冗余连接 II

在本问题中,有根树指满足以下条件的 有向 图。该树只有一个根节点,所有其他节点都是该根节点的后继。该树除了根节点之外的每一个节点都有且只有一个父节点,而根节点没有父节点。

输入一个有向图,该图由一个有着 n 个节点(节点值不重复,从 1 到 n)的树及一条附加的有向边构成。附加的边包含在 1 到 n 中的两个不同顶点间,这条附加的边不属于树中已存在的边。

结果图是一个以边组成的二维数组 edges 。 每个元素是一对 [ui, vi],用以表示 有向 图中连接顶点 ui 和顶点 vi 的边,其中 ui 是 vi 的一个父节点。

返回一条能删除的边,使得剩下的图是有 n 个节点的有根树。若有多个答案,返回最后出现在给定二维数组的答案。

示例 1:

在这里插入图片描述

输入:edges = [[1,2],[1,3],[2,3]]

输出:[2,3]

示例 2:

在这里插入图片描述

输入:edges = [[1,2],[2,3],[3,4],[4,1],[1,5]]

输出:[4,1]

提示:

n == edges.length

3 <= n <= 1000

edges[i].length == 2

1 <= ui, vi <= n

题解:
方法:并查集
        

class Solution {private int[] p;public int[] findRedundantDirectedConnection(int[][] edges) {int n = edges.length;int[] ind = new int[n];for (var e : edges) {++ind[e[1] - 1];}List<Integer> dup = new ArrayList<>();p = new int[n];for (int i = 0; i < n; ++i) {if (ind[edges[i][1] - 1] == 2) {dup.add(i);}p[i] = i;}if (!dup.isEmpty()) {for (int i = 0; i < n; ++i) {if (i == dup.get(1)) {continue;}int pu = find(edges[i][0] - 1);int pv = find(edges[i][1] - 1);if (pu == pv) {return edges[dup.get(0)];}p[pu] = pv;}return edges[dup.get(1)];}for (int i = 0;; ++i) {int pu = find(edges[i][0] - 1);int pv = find(edges[i][1] - 1);if (pu == pv) {return edges[i];}p[pu] = pv;}}private int find(int x) {if (p[x] != x) {p[x] = find(p[x]);}return p[x];}
}

29.生成不含相邻零的二进制字符串

题目链接:3211. 生成不含相邻零的二进制字符串

给你一个正整数 n。

如果一个二进制字符串 x 的所有长度为 2 的
子字符串
中包含 至少 一个 “1”,则称 x 是一个 有效 字符串。

返回所有长度为 n 的 有效 字符串,可以以任意顺序排列。

示例 1:

输入: n = 3

输出: [“010”,“011”,“101”,“110”,“111”]

解释:

长度为 3 的有效字符串有:“010”、“011”、“101”、“110” 和 “111”。

示例 2:

输入: n = 1

输出: [“0”,“1”]

解释:

长度为 1 的有效字符串有:“0” 和 “1”。

提示:

1 <= n <= 18

题解:
方法:回溯(爆搜)
        

class Solution {public List<String> validStrings(int n) {List<String> ans = new ArrayList<>();char[] path = new char[n];dfs(0, n, path, ans);return ans;}private void dfs(int i, int n, char[] path, List<String> ans) {if (i == n) {ans.add(new String(path));return;}// 填 1path[i] = '1';dfs(i + 1, n, path, ans);// 填 0if (i == 0 || path[i - 1] == '1') {path[i] = '0'; // 直接覆盖dfs(i + 1, n, path, ans);}}
}

30.交换后字典序最小的字符串

题目链接:3216. 交换后字典序最小的字符串

给你一个仅由数字组成的字符串 s,在最多交换一次 相邻 且具有相同 奇偶性 的数字后,返回可以得到的
字典序最小的字符串

如果两个数字都是奇数或都是偶数,则它们具有相同的奇偶性。例如,5 和 9、2 和 4 奇偶性相同,而 6 和 9 奇偶性不同。

示例 1:

输入: s = “45320”

输出: “43520”

解释:

s[1] == ‘5’ 和 s[2] == ‘3’ 都具有相同的奇偶性,交换它们可以得到字典序最小的字符串。

示例 2:

输入: s = “001”

输出: “001”

解释:

无需进行交换,因为 s 已经是字典序最小的。

提示:

2 <= s.length <= 100

s 仅由数字组成。

题解:
方法:贪心
        

class Solution {public String getSmallestString(String s) {char[] t = s.toCharArray();for (int i = 1; i < t.length; i++) {char x = t[i - 1];char y = t[i];if (x > y && x % 2 == y % 2) {t[i - 1] = y;t[i] = x;break;}}return new String(t);}
}

31.不包含相邻元素的子序列的最大和

题目链接:3165. 不包含相邻元素的子序列的最大和

给你一个整数数组 nums 和一个二维数组 queries,其中 queries[i] = [posi, xi]。

对于每个查询 i,首先将 nums[posi] 设置为 xi,然后计算查询 i 的答案,该答案为 nums 中 不包含相邻元素 的
子序列
的 最大 和。

返回所有查询的答案之和。

由于最终答案可能非常大,返回其对 109 + 7 取余 的结果。

子序列 是指从另一个数组中删除一些或不删除元素而不改变剩余元素顺序得到的数组。

示例 1:

输入:nums = [3,5,9], queries = [[1,-2],[0,-3]]

输出:21

解释:

执行第 1 个查询后,nums = [3,-2,9],不包含相邻元素的子序列的最大和为 3 + 9 = 12。

执行第 2 个查询后,nums = [-3,-2,9],不包含相邻元素的子序列的最大和为 9 。

示例 2:

输入:nums = [0,-1], queries = [[0,-5]]

输出:0

解释: 执行第 1 个查询后,nums = [-5,-1],不包含相邻元素的子序列的最大和为 0(选择空子序列)。

提示:

1 <= nums.length <= 5 * 104

-105 <= nums[i] <= 105

1 <= queries.length <= 5 * 104

queries[i] == [posi, xi]

0 <= posi <= nums.length - 1

-105 <= xi <= 105

题解:
方法:分治思想+线段树
        

class Solution {public int maximumSumSubsequence(int[] nums, int[][] queries) {int n = nums.length;// 4 个数分别保存 f00, f01, f10, f11long[][] t = new long[2 << (32 - Integer.numberOfLeadingZeros(n))][4];build(t, nums, 1, 0, n - 1);long ans = 0;for (int[] q : queries) {update(t, 1, 0, n - 1, q[0], q[1]);ans += t[1][3]; // 注意 f11 没有任何限制,也就是整个数组的打家劫舍}return (int) (ans % 1_000_000_007);}// 合并左右儿子private void maintain(long[][] t, int o) {long[] a = t[o * 2];long[] b = t[o * 2 + 1];t[o][0] = Math.max(a[0] + b[2], a[1] + b[0]);t[o][1] = Math.max(a[0] + b[3], a[1] + b[1]);t[o][2] = Math.max(a[2] + b[2], a[3] + b[0]);t[o][3] = Math.max(a[2] + b[3], a[3] + b[1]);}// 用 nums 初始化线段树private void build(long[][] t, int[] nums, int o, int l, int r) {if (l == r) {t[o][3] = Math.max(nums[l], 0);return;}int m = (l + r) / 2;build(t, nums, o * 2, l, m);build(t, nums, o * 2 + 1, m + 1, r);maintain(t, o);}// 把 nums[i] 改成 valprivate void update(long[][] t, int o, int l, int r, int i, int val) {if (l == r) {t[o][3] = Math.max(val, 0);return;}int m = (l + r) / 2;if (i <= m) {update(t, o * 2, l, m, i, val);} else {update(t, o * 2 + 1, m + 1, r, i, val);}maintain(t, o);}
}

2024.11

1.超级饮料的最大强化能量

题目链接:3259. 超级饮料的最大强化能量

来自未来的体育科学家给你两个整数数组 energyDrinkA 和 energyDrinkB,数组长度都等于 n。这两个数组分别代表 A、B 两种不同能量饮料每小时所能提供的强化能量。

你需要每小时饮用一种能量饮料来 最大化 你的总强化能量。然而,如果从一种能量饮料切换到另一种,你需要等待一小时来梳理身体的能量体系(在那个小时里你将不会获得任何强化能量)。

返回在接下来的 n 小时内你能获得的 最大 总强化能量。

注意 你可以选择从饮用任意一种能量饮料开始。

示例 1:

输入:energyDrinkA = [1,3,1], energyDrinkB = [3,1,1]

输出:5

解释:

要想获得 5 点强化能量,需要选择只饮用能量饮料 A(或者只饮用 B)。

示例 2:

输入:energyDrinkA = [4,1,1], energyDrinkB = [1,1,3]

输出:7

解释:

第一个小时饮用能量饮料 A。

切换到能量饮料 B ,在第二个小时无法获得强化能量。

第三个小时饮用能量饮料 B ,并获得强化能量。

提示:

n == energyDrinkA.length == energyDrinkB.length

3 <= n <= 105

1 <= energyDrinkA[i], energyDrinkB[i] <= 105

题解:
方法:递归搜索 + 保存递归返回值 = 记忆化搜索
        

class Solution {public long maxEnergyBoost(int[] a, int[] b) {int n = a.length;int[][] c = {a, b};long[][] memo = new long[n][2];return Math.max(dfs(n - 1, 0, c, memo), dfs(n - 1, 1, c, memo));}private long dfs(int i, int j, int[][] c, long[][] memo) {if (i < 0) {return 0;}if (memo[i][j] > 0) { // 之前计算过return memo[i][j];}return memo[i][j] = Math.max(dfs(i - 1, j, c, memo), dfs(i - 2, j ^ 1, c, memo)) + c[j][i];}
}

2.使两个整数相等的位更改次数

题目链接:3226. 使两个整数相等的位更改次数

给你两个正整数 n 和 k。

你可以选择 n 的 二进制表示 中任意一个值为 1 的位,并将其改为 0。

返回使得 n 等于 k 所需要的更改次数。如果无法实现,返回 -1。

示例 1:

输入: n = 13, k = 4

输出: 2

解释: 最初,n 和 k 的二进制表示分别为 n = (1101)2 和 k = (0100)2,

我们可以改变 n 的第一位和第四位。结果整数为 n = (0100)2 = k。

示例 2:

输入: n = 21, k = 21

输出: 0

解释: n 和 k 已经相等,因此不需要更改。

示例 3:

输入: n = 14, k = 13

输出: -1

解释: 无法使 n 等于 k。

提示:

1 <= n, k <= 106

题解:
方法:O(1) 位运算做法
        

class Solution {public int minChanges(int n, int k) {return (n & k) != k ? -1 : Integer.bitCount(n ^ k);}
}

3.大礼包

题目链接:638. 大礼包

在 LeetCode 商店中, 有 n 件在售的物品。每件物品都有对应的价格。然而,也有一些大礼包,每个大礼包以优惠的价格捆绑销售一组物品。

给你一个整数数组 price 表示物品价格,其中 price[i] 是第 i 件物品的价格。另有一个整数数组 needs 表示购物清单,其中 needs[i] 是需要购买第 i 件物品的数量。

还有一个数组 special 表示大礼包,special[i] 的长度为 n + 1 ,其中 special[i][j] 表示第 i 个大礼包中内含第 j 件物品的数量,且 special[i][n] (也就是数组中的最后一个整数)为第 i 个大礼包的价格。

返回 确切 满足购物清单所需花费的最低价格,你可以充分利用大礼包的优惠活动。你不能购买超出购物清单指定数量的物品,即使那样会降低整体价格。任意大礼包可无限次购买。

示例 1:

输入:price = [2,5], special = [[3,0,5],[1,2,10]], needs = [3,2]

输出:14

解释:有 A 和 B 两种物品,价格分别为 ¥2 和 ¥5 。

大礼包 1 ,你可以以 ¥5 的价格购买 3A 和 0B 。

大礼包 2 ,你可以以 ¥10 的价格购买 1A 和 2B 。

需要购买 3 个 A 和 2 个 B , 所以付 ¥10 购买 1A 和 2B(大礼包 2),以及 ¥4 购买 2A 。

示例 2:

输入:price = [2,3,4], special = [[1,1,0,4],[2,2,1,9]], needs = [1,2,1]

输出:11

解释:A ,B ,C 的价格分别为 ¥2 ,¥3 ,¥4 。

可以用 ¥4 购买 1A 和 1B ,也可以用 ¥9 购买 2A ,2B 和 1C 。

需要买 1A ,2B 和 1C ,所以付 ¥4 买 1A 和 1B(大礼包 1),以及 ¥3 购买 1B , ¥4 购买 1C 。

不可以购买超出待购清单的物品,尽管购买大礼包 2 更加便宜。

提示:

n == price.length == needs.length

1 <= n <= 6

0 <= price[i], needs[i] <= 10

1 <= special.length <= 100

special[i].length == n + 1

0 <= special[i][j] <= 50

生成的输入对于 0 <= j <= n - 1 至少有一个 special[i][j] 非零。

题解:
方法:状态压缩 + 记忆化搜索
        

class Solution {private final int bits = 4;private int n;private List<Integer> price;private List<List<Integer>> special;private Map<Integer, Integer> f = new HashMap<>();public int shoppingOffers(List<Integer> price, List<List<Integer>> special, List<Integer> needs) {n = needs.size();this.price = price;this.special = special;int mask = 0;for (int i = 0; i < n; ++i) {mask |= needs.get(i) << (i * bits);}return dfs(mask);}private int dfs(int cur) {if (f.containsKey(cur)) {return f.get(cur);}int ans = 0;for (int i = 0; i < n; ++i) {ans += price.get(i) * (cur >> (i * bits) & 0xf);}for (List<Integer> offer : special) {int nxt = cur;boolean ok = true;for (int j = 0; j < n; ++j) {if ((cur >> (j * bits) & 0xf) < offer.get(j)) {ok = false;break;}nxt -= offer.get(j) << (j * bits);}if (ok) {ans = Math.min(ans, offer.get(n) + dfs(nxt));}}f.put(cur, ans);return ans;}
}

下接:【题解】—— LeetCode一周小结45


http://www.lryc.cn/news/478067.html

相关文章:

  • faiss 用于检索10亿向量(维度768)的方法
  • sql专题 之 常用命令
  • Kubernetes Extended Resource 扩展资源使用简介
  • 基于STM32的天气时钟项目教学
  • 神经网络进行波士顿房价预测
  • C++builder中的人工智能(7)如何在C++中开发特别的AI激活函数?
  • 更改lvgl图片的分辨率(减少像素)达到减小内存占用的目的
  • python的socket库的基本使用总目录
  • golang学习3
  • Python解力扣算法题(六)(详解+注释)
  • 【C++】继承和多态常见的面试问题
  • 入门网络安全工程师要学习哪些内容(详细教程)
  • 【游戏引擎之路】登神长阶(十二)——DirectX11教程:If you‘re going through hell, keep going!
  • Python列表(一图秒了)
  • 雷池社区版 7.1.0 LTS 发布了
  • 推荐一款功能强大的数据库开发管理工具:SQLite Expert Pro
  • 动态规划 之 路径问题 算法专题
  • 从office套件接入GPT4谈自动化测试的前景
  • CentOS操作系统安装过程简介
  • 基于Multisim光控夜灯LED电路(含仿真和报告)
  • 导师双选系统开发:Spring Boot技术详解
  • 双11花了“一部手机钱”买手机壳的年轻人,究竟在买什么?
  • rediss数据结构及其底层实现
  • 自动化测试中使用Pytest Fixture?推荐10种常见用法!
  • Spring中的ConversionService,为Spring提供数据转换服务
  • gdb和make工具
  • 【d66】【Java】【力扣】174.寻找二叉搜索树中的目标节点
  • Spring Boot关闭时,如何确保内存里面的mq消息被消费完?
  • HTML 基础标签——文本内容标签 <ul>、<ol>、<blockquote> 、<code> 等标签的用法详解
  • 高效管理社团:Spring Boot在校园社团信息管理中的应用