第 120 场双周赛 解题报告 | 珂学家 | 前后缀拆解 启发式合并


前言

image.png

忘名可以再记,回忆永不再来


整体评价

好像有一段时间没写周赛题解了,_.

感觉今天手感特别好,下午的几场比赛,包括传智杯都能打出超神战绩。

T3这题属于前后缀拆解,然后单调栈上二分(可以引入哨兵机制),感觉单调栈不太严谨,写起来有点变扭。

T4难道是传说中Dsu On Tree? 感觉有些像。


T1. 统计移除递增子数组的数目 I

和T3一起讲


T2. 找到最大周长的多边形

思路:贪心

猜了一个结论

∑ j = 0 j = i a r r [ j ] > a r r [ i + 1 ] ,满足此条件的最大 i \sum_{j=0}^{j=i} arr[j] > arr[i+1], 满足此条件的最大i j=0j=iarr[j]>arr[i+1],满足此条件的最大i

先对 a r r arr arr排序,逆序找到第一个 i i i即可

class Solution {public long largestPerimeter(int[] nums) {// 思维题long sum = 0;Arrays.sort(nums);for (int i = 0; i < nums.length; i++) {sum += nums[i];}// 逆序for (int i = nums.length - 1; i >= 2; i--) {sum -= nums[i];if (sum > nums[i]) {return sum + nums[i];}}return -1;}}

T3. 统计移除递增子数组的数目 II

思路: 前后缀拆解 + 单调栈上二分

因为题目要求最左侧和最右侧都严格递增,所以需要预处理前后缀,保证严格递增

从左往右枚举每个点v

  • check后缀是递增的
  • 寻找前缀构建的单调栈,且结尾小于v的点,累加数量
  • 如果当前值前缀是递增的,则加入单调栈
class Solution {public long incremovableSubarrayCount(int[] nums) {int n = nums.length;boolean[] pre = new boolean[n];boolean[] suf = new boolean[n];pre[0] = suf[n - 1] = true;for (int i = 1; i < n; i++) {pre[i] = pre[i - 1] && nums[i] > nums[i - 1];}for (int i = n - 2; i >= 0; i--) {suf[i] = suf[i + 1] && nums[i] < nums[i + 1];}long res = 0;// java可以用treemap来偷鸡单调栈monostackTreeMap<Integer, Integer> range = new TreeMap<>();range.put(-1, 1); // 哨兵for (int i = 0; i < n; i++) {int v = nums[i];if (suf[i]) {var ent = range.lowerEntry(v);if (ent != null) {// 删除的子数组必要有1个元素,所以要分类讨论if (ent.getValue() + 1 == i + 2) {res += ent.getValue() - 1;} else {res += ent.getValue();}}}if (pre[i]) {// 为啥要+2, 主要是为了统计方便range.put(v, i + 2);}}// 处理尾巴{var ent = range.lowerEntry(Integer.MAX_VALUE);if (ent != null) {if (ent.getValue() + 1 == n + 2) {res += ent.getValue() - 1;} else {res += ent.getValue();}}}return res;}}

T4. 树中每个节点放置的金币数目

思路: 启发式合并

有一个结论:

如果一个序列 a r r , a r r [ 0 ] , a r r [ 1 ] , . . . , , a r r [ n − 2 ] , a r r [ n − 1 ] , 抽取其中 3 个数使其乘积最大 如果一个序列arr, arr[0], arr[1], ..., , arr[n - 2], arr[n - 1], 抽取其中3个数使其乘积最大 如果一个序列arr,arr[0],arr[1],...,,arr[n2],arr[n1],抽取其中3个数使其乘积最大

取 a r r 的最小 3 个数, a 1 , a 2 , a 3 ( a 1 ≤ a 2 ≤ a 3 ) 取arr的最小3个数, a_1, a_2, a_3 (a_1 \le a_2 \le a_3) arr的最小3个数,a1,a2,a3(a1a2a3)

最大的 3 个数 , b 1 , b 2 , b 3 ( b 1 ≤ b 2 ≤ b 3 ) 最大的3个数, b_1, b_2, b_3 (b_1 \le b_2 \le b_3) 最大的3个数,b1,b2,b3(b1b2b3)

乘积最大 = m a x ( a 1 ∗ a 2 ∗ a 3 , a 1 ∗ a 2 ∗ b 3 , a 1 ∗ b 2 ∗ b 3 , b 1 ∗ b 2 ∗ b 3 ) 乘积最大 = max(a_1 * a_2 * a_3, a_1 * a_2 * b_3, a_1 * b_2 * b_3, b_1 * b_2 * b_3) 乘积最大=max(a1a2a3,a1a2b3,a1b2b3,b1b2b3)

有了这个结论后,剩下的就好办了

每个子节点再往上传的时候,只需要保留3个最小数,3个最大数即可。

而这点,就扣合本题的思路

启发式合并 启发式合并 启发式合并

class Solution {int n;List<Integer>[]g;int[] cost;long[] res;List<Integer> []mins;List<Integer> []maxs;void dfs(int u, int fa) {List<Integer> tmpMin = new ArrayList<>();List<Integer> tmpMax = new ArrayList<>();tmpMin.add(cost[u]);tmpMax.add(cost[u]);for (int v: g[u]) {if (v == fa) continue;dfs(v, u);for (int tv: mins[v]) {tmpMin.add(tv);}for (int tv: maxs[v]) {tmpMax.add(tv);}}if (tmpMin.size() < 3) {res[u] = 1;} else {Collections.sort(tmpMin);Collections.sort(tmpMax);// 核心逻辑long ans = Long.MIN_VALUE / 10;long a1 = tmpMin.get(0), a2 = tmpMin.get(1), a3 = tmpMin.get(2);int nz = tmpMax.size();long a4 = tmpMax.get(nz - 3), a5 = tmpMax.get(nz - 2), a6 = tmpMax.get(nz - 1);ans = Math.max(ans, a1 * a2 * a3);ans = Math.max(ans, a1 * a2 * a6);ans = Math.max(ans, a1 * a5 * a6);ans = Math.max(ans, a4 * a5 * a6);if (ans < 0) {res[u] = 0;} else {res[u] = ans;}}// 保留3位,往上传for (int i = 0; i < 3 && i < tmpMin.size(); i++) {mins[u].add(tmpMin.get(i));}for (int i = tmpMax.size() - 1; i >= 0 && tmpMax.size() - i <= 3; i--) {maxs[u].add(tmpMax.get(i));}}public long[] placedCoins(int[][] edges, int[] cost) {n = cost.length;g = new List[n];this.cost = cost;Arrays.setAll(g, x->new ArrayList<>());for (int[] e: edges) {g[e[0]].add(e[1]);g[e[1]].add(e[0]);}mins = new List[n];Arrays.setAll(mins, x->new ArrayList<>());maxs = new List[n];Arrays.setAll(maxs, x->new ArrayList<>());res = new long[n];dfs(0, -1);return res;}}

写在最后

即使是希望、即使是梦想,都是需要被守护的。

image.png

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/296376.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

SpringSecurity深度解析与实践(3)

这里写自定义目录标题 引言SpringSecurity之授权授权介绍java权限集成 登录失败三次用户上锁 引言 SpringSecurity深度解析与实践&#xff08;2&#xff09;的网址 SpringSecurity之授权 授权介绍 Spring Security 中的授权分为两种类型&#xff1a; 基于角色的授权&#…

5. 结构型模式 - 外观模式

亦称&#xff1a; Facade 意图 外观模式是一种结构型设计模式&#xff0c; 能为程序库、 框架或其他复杂类提供一个简单的接口 问题 假设你必须在代码中使用某个复杂的库或框架中的众多对象。 正常情况下&#xff0c; 你需要负责所有对象的初始化工作、 管理其依赖关系并按正确…

HackTheBox - Medium - Linux - Format

Format Format 是一种中等难度的 Linux 机器&#xff0c;它突出显示了由解决方案的结构方式引起的安全问题。立足点涉及PHP源代码审查&#xff0c;发现和利用本地文件读/写漏洞&#xff0c;并利用Nginx中的错误配置在Redis Unix套接字上执行命令。横向移动包括浏览 Redis 数据…

饥荒Mod 开发(二一):超大便携背包,超大物品栏,永久保鲜

饥荒Mod 开发(二十)&#xff1a;显示打怪伤害值 源码 游戏中的物品栏容量实在太小了&#xff0c;虽然可以放在箱子里面但是真的很不方便&#xff0c;外出一趟不容易看到东西都不能捡。实在是虐心。 游戏中的食物还有变质机制&#xff0c;时间长了就不能吃了&#xff0c;玩这个游…

L1-060:心理阴影面积

题目描述 这是一幅心理阴影面积图。我们都以为自己可以匀速前进&#xff08;图中蓝色直线&#xff09;&#xff0c;而拖延症晚期的我们往往执行的是最后时刻的疯狂赶工&#xff08;图中的红色折线&#xff09;。由红、蓝线围出的面积&#xff0c;就是我们在做作业时的心理阴影面…

基于 Editor.js 开发富文本编辑器库

开始 Editor.js 提供了简单而直观的用户界面&#xff0c;根据需求可以灵活添加自定义的编辑工具&#xff0c;通过插件扩展功能 Editorjs 使用 js 开发&#xff0c;脱离框架依赖&#xff0c;因此可以基于它封装富文本编辑器&#xff0c;用于 Vue 和 React 项目 editor-js-com…

计算机网络复习-OSI TCP/IP 物理层

我膨胀了&#xff0c;挂我啊~ 作者简介&#xff1a; 每年都吐槽吉师网安奇怪的课程安排、全校正经学网络安全不超20人情景以及割韭菜企业合作的FW&#xff0c;今年是第一年。。 TCP/IP模型 先做两道题&#xff1a; TCP/IP协议模型由高层到低层分为哪几层&#xff1a; 这题…

【数据库系统概论】第2章-关系数据库

复习记录 2.1 关系数据结构及形式化定义2.1.1 关系2.1.2 关系模式2.1.3 关系数据库 2.2 关系操作2.3 关系的完整性2.4 关系代数题目 2.1 关系数据结构及形式化定义 2.1.1 关系 一些概念 关系 R ( D 1 , D 2 , . . . , D n ) R(D_1,D_2,...,D_n) R(D1​,D2​,...,Dn​) D i…

深度学习入门(python)考试速成均方误差

均方误差 表示神经网络的输出&#xff0c;表示监督数据&#xff0c;表示数据的维度。 这里神经网络的输出y是softmax函数的输出 数组元素的索引从第一个开始依次对应数组“0”&#xff0c;“1”&#xff0c;“2”&#xff0c;...... 由于softmax函数的输出可理解为概率 由此…

css radial-gradient 径向渐变基本语法与使用

在之前的文章《深入理解Css linear-gradient线性渐变》我们了解了CSS中的线性渐变&#xff0c;本文将介绍CSS中的另一种渐变———径向渐变&#xff08;Radial Gradient&#xff09;&#xff1a; CSS中的径向渐变&#xff08;Radial Gradient&#xff09;允许你创建从一个颜色…

【计数DP】牛客小白月赛19

登录—专业IT笔试面试备考平台_牛客网 题意 思路 首先做法一定是计数 dp 然后状态设计&#xff0c;先设 dp[i] 然后看影响决策的因素&#xff1a;两边的火焰情况&#xff0c;那就 dp[i][0/1][0/1]表示 前 i 个&#xff0c;该位有无火焰&#xff0c;该位右边有无火焰的方案数…

redis基本用法学习(C#调用CSRedisCore操作redis)

除了NRedisStack包&#xff0c;csredis也是常用的redis操作模块&#xff08;从EasyCaching提供的常用redis操作包来看&#xff0c;CSRedis、freeredis、StackExchange.Redis应该都属于常用redis操作模块&#xff09;&#xff0c;本文学习使用C#调用CSRedis包操作redis的基本方式…