【力扣周赛】第 369 场周赛(⭐记忆化搜索 树形DP)

2023-12-02 05:44

本文主要是介绍【力扣周赛】第 369 场周赛(⭐记忆化搜索 树形DP),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 竞赛链接
  • Q1:2917. 找出数组中的 K-or 值
    • 竞赛时代码——按题意模拟
  • Q2:2918. 数组的最小相等和
    • 竞赛时代码——分类讨论
  • Q3:2919. 使数组变美的最小增量运算数⭐⭐⭐
    • 竞赛时代码——动态规划
    • 解法2——记忆化搜索 翻译成递推
  • Q4:2920. 收集所有金币可获得的最大积分⭐⭐⭐⭐⭐🐂
    • 解法1——自顶向下(记忆化搜索)
    • 解法2——自底向上
  • 成绩记录

竞赛链接

https://leetcode.cn/contest/weekly-contest-369/

Q1:2917. 找出数组中的 K-or 值

https://leetcode.cn/problems/find-the-k-or-of-an-array/description/

在这里插入图片描述

提示:
1 <= nums.length <= 50
0 <= nums[i] < 2^31
1 <= k <= nums.length

竞赛时代码——按题意模拟

class Solution {public int findKOr(int[] nums, int k) {int ans = 0;for (int i = 0; i < 31; ++i) {int c = 0;for (int x: nums) {if ((x >> i & 1) == 1) c++;}if (c >= k) ans |= 1 << i;}return ans;}
}

Q2:2918. 数组的最小相等和

https://leetcode.cn/problems/minimum-equal-sum-of-two-arrays-after-replacing-zeros/description/

在这里插入图片描述

提示:
1 <= nums1.length, nums2.length <= 10^5
0 <= nums1[i], nums2[i] <= 10^6

竞赛时代码——分类讨论

在这里插入图片描述

class Solution {public long minSum(int[] nums1, int[] nums2) {long s1 = 0, s2 = 0;int cnt1 = 0, cnt2 = 0;for (int x: nums1) {s1 += x;if (x == 0) cnt1++;}for (int x: nums2) {s2 += x;if (x == 0) cnt2++;}if (s1 < s2 + cnt2 && cnt1 == 0) return -1;if (s1 + cnt1 > s2 && cnt2 == 0) return -1;return Math.max(s1 + cnt1, s2 + cnt2);}
}

Q3:2919. 使数组变美的最小增量运算数⭐⭐⭐

https://leetcode.cn/problems/minimum-increment-operations-to-make-array-beautiful/description/

在这里插入图片描述

提示:
3 <= n == nums.length <= 10^5
0 <= nums[i] <= 10^9
0 <= k <= 10^9

竞赛时代码——动态规划

写的有点丑陋,但好歹是过了。

class Solution {public long minIncrementOperations(int[] nums, int k) {int n = nums.length;long[][] dp = new long[n + 1][2];       // dp[i][j]表示到第i个位置,选或不选且满足条件时的最小增量for (int i = 0; i <= n; ++i) Arrays.fill(dp[i], Long.MAX_VALUE);dp[0][0] = 0;dp[1][1] = Math.max(0, k - nums[0]);for (int i = 1; i <= n; ++i) {long x = nums[i - 1];if (i - 1 >= 0) {       // 前一个dp[i][0] = Math.min(dp[i][0], dp[i - 1][1]);dp[i][1] = Math.min(dp[i][1], Math.min(dp[i - 1][0], dp[i - 1][1]) + Math.max(0, k - x));}if (i - 2 >= 0) {       // 前两个dp[i][0] = Math.min(dp[i][0], dp[i - 2][1]);if (i - 3 >= 0) dp[i][1] = Math.min(dp[i][1], Math.min(dp[i - 2][1], i - 3 == 0? 0: dp[i - 3][1]) + Math.max(0, k - x));else dp[i][1] = Math.min(dp[i][1], Math.min(dp[i - 2][1], dp[i - 2][0]) + Math.max(0, k - x));}}if (dp[n][0] > dp[n][1]) return dp[n][1];return dp[n][0];}
}

解法2——记忆化搜索 翻译成递推

https://leetcode.cn/problems/minimum-increment-operations-to-make-array-beautiful/solutions/2503157/qiao-miao-she-ji-zhuang-tai-xuan-huo-bu-8547u/
在这里插入图片描述

dfs(i,j)表示前0~i个数字,且后面有j个不到k的数字,此时的最小花费

class Solution {int n, k;int[] nums;long[][] memo;public long minIncrementOperations(int[] nums, int k) {n = nums.length;this.k = k;this.nums = nums;memo = new long[n][3];for (long[] m: memo) Arrays.fill(m, -1);return dfs(n - 1, 0);}// dfs(i,j)表示前0~i个数字,且后面有j个不到k的数字,此时的最小花费public long dfs(int i, int j) {if (i < 0) return 0;if (memo[i][j] != -1) return memo[i][j];long res = dfs(i - 1, 0) + Math.max(0, k - nums[i]);    // 增加当前位置if (j < 2) res = Math.min(res, dfs(i - 1, j + 1));      // j<2时,可以不增加当前位置return memo[i][j] = res;}
}

1:1翻译成递推

自己改成了下面这样子。

class Solution {public long minIncrementOperations(int[] nums, int k) {int n = nums.length;// dp[i][j]表示到下标i,前面有j个位置没有到k(包括当前位置)long[][] dp = new long[n + 1][3];for (int i = 0; i < n; ++i) {for (int j = 0; j < 3; ++j) {dp[i + 1][j] = dp[i][2] + Math.max(0, k - nums[i]);     // 不得不选if (j > 0) {dp[i + 1][j] = Math.min(dp[i + 1][j], dp[i][j - 1]);// 可以不选}}}return Arrays.stream(dp[n]).min().getAsLong();}
}

Q4:2920. 收集所有金币可获得的最大积分⭐⭐⭐⭐⭐🐂

https://leetcode.cn/problems/maximum-points-after-collecting-coins-from-all-nodes/description/

在这里插入图片描述

在这里插入图片描述

解法1——自顶向下(记忆化搜索)

https://leetcode.cn/problems/maximum-points-after-collecting-coins-from-all-nodes/solutions/2503152/shu-xing-dp-ji-yi-hua-sou-suo-by-endless-phzx/
在这里插入图片描述

class Solution {List<Integer>[] g;int[] coins;int k;int[][] memo;public int maximumPoints(int[][] edges, int[] coins, int k) {int n = edges.length + 1;this.coins = coins;this.k = k;g = new ArrayList[n];Arrays.setAll(g, e -> new ArrayList<>());for (int[] e: edges) {g[e[0]].add(e[1]);g[e[1]].add(e[0]);}memo = new int[n][14];      // 该题目数据下最多右移14次变成0for (int i = 0; i < n; ++i) {Arrays.fill(memo[i], -1);}return dfs(0, -1, 0);}public int dfs(int x, int fa, int d) {if (memo[x][d] != -1) return memo[x][d];int res1 = (coins[x] >> d) - k;int res2 = coins[x] >> (d + 1);for (int y: g[x]) {if (y == fa) continue;res1 += dfs(y, x, d);if (d < 13) {res2 += dfs(y, x, d + 1);}}return memo[x][d] = Math.max(res1, res2);}
}

解法2——自底向上

自底向上每个节点都只会枚举一遍,不需要 记忆数组了。

class Solution {List<Integer>[] g;int[] coins;int k;public int maximumPoints(int[][] edges, int[] coins, int k) {int n = edges.length + 1;this.coins = coins;this.k = k;g = new ArrayList[n];Arrays.setAll(g, e -> new ArrayList<>());for (int[] e: edges) {g[e[0]].add(e[1]);g[e[1]].add(e[0]);}return dfs(0, -1)[0];}public int[] dfs(int x, int fa) {int[] res1 = new int[14], res2 = new int[14];// 先得到各个子节点的结果for (int y: g[x]) {if (y != fa) {int[] r = dfs(y, x);for (int j = 0; j < 14; ++j) {res1[j] += r[j];if (j < 13) res2[j] += r[j + 1];}}}// 再逐个处理for (int j = 0; j < 14; ++j) {res1[j] = Math.max(res1[j] + (coins[x] >> j) - k, res2[j] + (coins[x] >> (j + 1)));}return res1;}
}

成绩记录

在这里插入图片描述
多的不说,排名挺吉利的。。
无奈遗憾掉分。
在这里插入图片描述

这篇关于【力扣周赛】第 369 场周赛(⭐记忆化搜索 树形DP)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/444168

相关文章

力扣SQL50 每位经理的下属员工数量 join

Problem: 1731. 每位经理的下属员工数量 👨‍🏫 参考题解 Code select m.Employee_id, m.name,count(*) reports_count,round(avg(e.age),0) average_agefrom Employees ejoin Employees mon e.reports_to = m.Employee_id

【文末附gpt升级秘笈】腾讯元宝AI搜索解析能力升级:千万字超长文处理的新里程碑

腾讯元宝AI搜索解析能力升级:千万字超长文处理的新里程碑 一、引言 随着人工智能技术的飞速发展,自然语言处理(NLP)和机器学习(ML)在各行各业的应用日益广泛。其中,AI搜索解析能力作为信息检索和知识抽取的核心技术,受到了广泛的关注和研究。腾讯作为互联网行业的领军企业,其在AI领域的探索和创新一直走在前列。近日,腾讯旗下的AI大模型应用——腾讯元宝,迎来了1.1.7版本的升级,新版本在AI搜

代码随想录算法训练营第三十九天|62.不同路径 63. 不同路径 II 343.整数拆分 96.不同的二叉搜索树

LeetCode 62.不同路径 题目链接:62.不同路径 踩坑:二维的vector数组需要初始化,否则会报错访问空指针 思路: 确定动态数组的含义:dp[i][j]:到达(i,j)有多少条路经递推公式:dp[i][j] = dp[i-1][j] + dp[i][j-1]初始化动态数组:dp[0][0] = 1遍历顺序:从左到右,从上到下 代码: class Solution {pu

上海邀请赛 A题目 HDU 5236(dp)

先求出没有ctrl+s的时候构造长度为i的期望f[i] 。然后枚举保存的次数,求出最小即可。 #include<cstdio>#include<cstdio>#include<cmath>#include<queue>#include<stack>#include<string>#include<cstring>#include<iostream>#include<map>

poj 3160 Father Christmas flymouse 强连通+dp

首先我们可以确定的是,对于val值小于0的节点都变成0.   假设一个集合内2个房间都能任意到达,那么我就可以吧集合内的所有点的价值都取到,并且可以达到任一点。实际上集合内的每个点是相同的,这样的集合就是一个强连通分量。 那么我们就可以用tarjin算法进行强连通缩点, 最后形成一个dag的图。在dag的图上面进行dp。可以先用拓扑排序后dp。或者建反响边记忆化搜索 。 VIEW

秋招突击——6/22——复习{区间DP——加分二叉树,背包问题——买书}——新作{移除元素、实现strStr()}

文章目录 引言复习区间DP——加分二叉树个人实现 背包问题——买书个人实现参考实现 新作移除元素个人实现参考思路 找出字符串中第一个匹配项的下标个人实现参考实现 总结 引言 今天做了一个噩梦,然后流了一身汗,然后没起来,九点多才起床背书。十点钟才开始把昨天那道题题目过一遍,然后十一点才开始复习题目,为了不耽误下午的时间,所以这里的就单纯做已经做过的题目,主打一个有量,不在学

leetcode刷题(45)——35. 二叉搜索树的最近公共祖先

给定一个二叉搜索树, 找到该树中两个指定节点的最近公共祖先。 百度百科中最近公共祖先的定义为:“对于有根树 T 的两个结点 p、q,最近公共祖先表示为一个结点 x,满足 x 是 p、q 的祖先且 x 的深度尽可能大(一个节点也可以是它自己的祖先)。” 例如,给定如下二叉搜索树: root = [6,2,8,0,4,7,9,null,null,3,5] 示例 1: 输入: root = [

力扣SQL50 游戏玩法分析 IV 子查询

Problem: 550. 游戏玩法分析 IV 👨‍🏫 参考题解 这个SQL查询的目的是计算每个玩家在登录后的第二天参与活动的比例。查询使用了子查询和左连接来实现这一目的。下面是查询的详细解释,包括每个部分的作用和注释: -- 计算每个玩家登录后第二天参与活动的比例select round(avg(a.event_date is not null), 2) as fractio

【智能优化算法改进策略之局部搜索算子(五)—自适应Rosenbrock坐标轮换法】

1、原理介绍 作为一种有效的直接搜索技术,Rosenbrock坐标轮换法[1,2]是根据Rosenbrock著名的“香蕉函数”的特点量身定制的,该函数的最小值位于曲线狭窄的山谷中。此外,该方法是一种典型的基于自适应搜索方向集的无导数局部搜索技术。此法于1960年由Rosenbrock提出,它与Hooke-Jeeves模式搜索法有些类似,但比模式搜索更为有效。每次迭代运算分为两部分[3]: 1)

Day59 代码随想录打卡|二叉树篇---把二叉搜索树转换为累加树

题目(leecode T538): 给出二叉 搜索 树的根节点,该树的节点值各不相同,请你将其转换为累加树(Greater Sum Tree),使每个节点 node 的新值等于原树中大于或等于 node.val 的值之和。 提醒一下,二叉搜索树满足下列约束条件: 节点的左子树仅包含键 小于 节点键的节点。节点的右子树仅包含键 大于 节点键的节点。左右子树也必须是二叉搜索树。 方法:本题