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

2023-12-24 14:28

本文主要是介绍第 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

这篇关于第 120 场双周赛 解题报告 | 珂学家 | 前后缀拆解 启发式合并的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

hdu2241(二分+合并数组)

题意:判断是否存在a+b+c = x,a,b,c分别属于集合A,B,C 如果用暴力会超时,所以这里用到了数组合并,将b,c数组合并成d,d数组存的是b,c数组元素的和,然后对d数组进行二分就可以了 代码如下(附注释): #include<iostream>#include<algorithm>#include<cstring>#include<stack>#include<que

【专题】2024飞行汽车技术全景报告合集PDF分享(附原数据表)

原文链接: https://tecdat.cn/?p=37628 6月16日,小鹏汇天旅航者X2在北京大兴国际机场临空经济区完成首飞,这也是小鹏汇天的产品在京津冀地区进行的首次飞行。小鹏汇天方面还表示,公司准备量产,并计划今年四季度开启预售小鹏汇天分体式飞行汽车,探索分体式飞行汽车城际通勤。阅读原文,获取专题报告合集全文,解锁文末271份飞行汽车相关行业研究报告。 据悉,业内人士对飞行汽车行业

day-51 合并零之间的节点

思路 直接遍历链表即可,遇到val=0跳过,val非零则加在一起,最后返回即可 解题过程 返回链表可以有头结点,方便插入,返回head.next Code /*** Definition for singly-linked list.* public class ListNode {* int val;* ListNode next;* ListNode() {}*

计算机毕业设计 大学志愿填报系统 Java+SpringBoot+Vue 前后端分离 文档报告 代码讲解 安装调试

🍊作者:计算机编程-吉哥 🍊简介:专业从事JavaWeb程序开发,微信小程序开发,定制化项目、 源码、代码讲解、文档撰写、ppt制作。做自己喜欢的事,生活就是快乐的。 🍊心愿:点赞 👍 收藏 ⭐评论 📝 🍅 文末获取源码联系 👇🏻 精彩专栏推荐订阅 👇🏻 不然下次找不到哟~Java毕业设计项目~热门选题推荐《1000套》 目录 1.技术选型 2.开发工具 3.功能

【每日一题】LeetCode 2181.合并零之间的节点(链表、模拟)

【每日一题】LeetCode 2181.合并零之间的节点(链表、模拟) 题目描述 给定一个链表,链表中的每个节点代表一个整数。链表中的整数由 0 分隔开,表示不同的区间。链表的开始和结束节点的值都为 0。任务是将每两个相邻的 0 之间的所有节点合并成一个节点,新节点的值为原区间内所有节点值的和。合并后,需要移除所有的 0,并返回修改后的链表头节点。 思路分析 初始化:创建一个虚拟头节点

Python:豆瓣电影商业数据分析-爬取全数据【附带爬虫豆瓣,数据处理过程,数据分析,可视化,以及完整PPT报告】

**爬取豆瓣电影信息,分析近年电影行业的发展情况** 本文是完整的数据分析展现,代码有完整版,包含豆瓣电影爬取的具体方式【附带爬虫豆瓣,数据处理过程,数据分析,可视化,以及完整PPT报告】   最近MBA在学习《商业数据分析》,大实训作业给了数据要进行数据分析,所以先拿豆瓣电影练练手,网络上爬取豆瓣电影TOP250较多,但对于豆瓣电影全数据的爬取教程很少,所以我自己做一版。 目

开题报告中的研究方法设计:AI能帮你做什么?

AIPaperGPT,论文写作神器~ https://www.aipapergpt.com/ 大家都准备开题报告了吗?研究方法部分是不是已经让你头疼到抓狂? 别急,这可是大多数人都会遇到的难题!尤其是研究方法设计这一块,选定性还是定量,怎么搞才能符合老师的要求? 每次到这儿,头脑一片空白。 好消息是,现在AI工具火得一塌糊涂,比如ChatGPT,居然能帮你在研究方法这块儿上出点主意。是不

【干货分享】基于SSM的体育场管理系统的开题报告(附源码下载地址)

中秋送好礼 中秋佳节将至,祝福大家中秋快乐,阖家幸福。本期免费分享毕业设计作品:《基于SSM的体育场管理系统》。 基于SSM的体育场管理系统的开题报告 一、课题背景与意义 随着全民健身理念的深入人心,体育场已成为广大师生和社区居民进行体育锻炼的重要场所。然而,传统的体育场管理方式存在诸多问题,如资源分配不均、预约流程繁琐、数据统计不准确等,严重影响了体育场的使用效率和用户体验。

【Python从入门到进阶】64、Pandas如何实现数据的Concat合并

接上篇《63.Pandas如何实现数据的Merge》 上一篇我们学习了Pandas如何实现数据的Merge,本篇我们来继续学习Pandas如何实现数据的Concat合并。 一、引言 在数据处理过程中,经常需要将多个数据集合并为一个统一的数据集,以便进行进一步的分析或建模。这种需求在多种场景下都非常常见,比如合并不同来源的数据集以获取更全面的信息、将时间序列数据按时间顺序拼接起来以观察长期趋势等

【QNX+Android虚拟化方案】120 - Android 侧 USB2.0 插拔过程

【QNX+Android虚拟化方案】120 - Android 侧 USB2.0 插拔过程 基于原生纯净代码,自学总结 纯技术分享,不会也不敢涉项目、不泄密、不传播代码文档!!! 本文禁止转载分享 !!! 汇总链接:《【QNX+Android虚拟化方案】00 - 系列文章链接汇总》 本文链接:《【QNX+Android虚拟化方案】120 - Android 侧 USB2.0