AT2346 [ARC070B] No Need
这里介绍几种做法。
解法一
首先可以转化一下题意,\(\forall i\) 如果 \(i\) 不是可有可无的当且仅当不用 \(i\) 能拼出 \([K - a_i, K)\) 中的数。
基于观察可以发现,对于 \(\forall i\) 如果 \(i\) 不是可有可无的,那么 \(\forall j, a_j \ge a_i\) 都不是可有可无的,证明如下:
取出使得对于 \(i\) 能拼出 \([K - a_i, K)\) 中一个数的集合 \(S\),分以下两种情况讨论:
若 \(j \in S\),考虑用 \(a_i\) 替换 \(a_j\) 使得对于 \(j\) 合法。
则可知 \(K - a_i \le |S| \le K \Rightarrow K \le |S| + a_i \le K - a_i\)
于是有 \(K - a_j \le |S| + a_i - a_j \le K + a_i - a_j \le K\) 因此替换是可行的。
若 \(j \notin S\) 直接使用 \(S\) 即可满足 \(j\) 的要求。
因此我们先按照 \(a_i\) 从小到大排序,可替换的必然是一段前缀,直接二分加背包判断即可,复杂度 \(O(nk \log n)\)。
解法二
一个直接的想法在于求出不使用 \(i\) 能拼成的数字有哪些。
暴力的做法就是 \(\forall i\) 枚举 \(j \ne i\) 进行暴力 \(dp\)。
但是你会发现有对于一些 \(i\) 有很多 \(j\) 都是能枚举的,换而言之每一对点对 \((i, j)(i < j)\) 都会在 \(i, j\) 上分别用对方 \(dp\) 一次。
对于这种点对对答案的贡献问题,经常可以考虑使用 \(\rm CDQ\) 分治解决。
那么我们在分治 \([l, r]\) 这个区间时可以发现,\([l, mid]\) 内的所有数会对 \([mid + 1, r]\) 内所有数的 \(dp\) 数组造成贡献。
因此我们直接考虑枚举左区间的每个数 \(i\) 让其在右区间恰好 \(dp\) 一次,对于右区间同理。
那么分治下去的时候就要同时将整个区间的 \(dp\) 数组传递下去,不难发现这样的复杂度实际上只枚举了 \(n \log n\) 个点,因此复杂度是 \(O(nk \log n)\) 的。
解法三
注意到我们要求的实质上是去除掉 \(i\) 以后的 \(dp\) 数组,不难让我们想起了 消失之物 这题的退背包做法。
但是本题的 \(dp\) 是判定性的,不存在可加性,怎么办呢?
不难发现只需要改写 \(dp\) 状态让其满足可加性即可,即我们将存储的值由是否可行变为方案数。
那么一个数不能被拼出来当且仅当方案数为 \(0\)。
于是先 \(dp\) 出总方案数,然后从左至右做退背包即可。
由于方案数过大,需要取模,复杂度 \(O(nk)\)。
解法四
考虑另一个暴力合并 \(dp\) 数组的做法,先记录 \(pre, suf\) 两个前缀/后缀 \(dp\) 数组。
\(\forall i\),在 \(i\) 处暴力合并 \(pre_{i - 1}, suf_{i + 1}\),复杂度 \(O(nk ^ 2)\)。
但实际上我们并不关心具体的 \(dp\) 数组,我们只关心 \([K - a_i, K)\) 之间有没有一个数能被拼出来。
为了尽量满足存在这样一个数,我们这样来看待这个问题:在满足和小于 \(K\) 的情况下最大化总和。
因此在合并的时候我们从小到大枚举 \(pre_{i - 1}\) 中一个可行的数 \(j\),找到 \(suf_{i + 1}\) 中满足 \(j + k < K\) 的最大的 \(k\)。
不难发现在 \(j\) 从小到大枚举的过程中,\(k\) 单调不升,因此这样寻找的复杂度就变为了 \(O(nk)\)。
对于解法一,其精髓在于通过观察获得的题目优秀性质。
对于解法二,其精髓在于将问题转化成点对之间的贡献问题,可以使用 \(\rm CDQ\) 分治优化。
对于解法三,其精髓在于将判定性问题转化成计数问题,满足可加性后就可以退背包。
对于解法四,其精髓在于去除了无用的一些状态,只寻找最用状态以降低复杂度。
四种做法各有优劣,但除做法二外其余的做法具有较大的局限性。
但可以发现的是,做法二可以解决很大一部分去除掉某个点后整体能够到达的状态,是一个通用的 \(\rm trick\)。
AT2346 [ARC070B] No Need的更多相关文章
- AT2346 No Need
atcoder上的题目 链接 一道思维题目 可以发现如果X是可有可无的,那么所有小于X的数也一定是可有可无的, 所有我们只要找出最大的那个可有可无的数字就好了 进一步分析,发现 若A1, A2, . ...
随机推荐
- Spring MVC 文件上传、Restful、表单校验框架
目录 文件上传 Restful Restful 简介 Rest 行为常用约定方式 Restful开发入门 表单校验框架 表单校验框架介绍 快速入门 多规则校验 嵌套校验 分组校验 综合案例 实用校验范 ...
- HTML网页设计基础笔记 • 【第5章 常用的样式属性】
全部章节 >>>> 本章目录 5.1 字体及文本属性 5.1.1 字体属性 5.1.2 文本属性 5.2 边距和填充 5.2.1 边距 5.2.2 填充 5.3 边框属性 ...
- Java初学者作业——定义管理员类(Admin),管理员类中的属性包括:姓名、账号、密码、电话;方法包括:登录、显示自己的信息。
返回本章节 返回作业目录 需求说明: 定义管理员类(Admin),管理员类中的属性包括:姓名.账号.密码.电话:方法包括:登录.显示自己的信息. 实现思路: 分析类的属性及其变量类型. 分析类的方法及 ...
- 编写Java程序,实现客户端向服务端上传文件的功能
查看本章节 查看作业目录 需求说明: 实现客户端向服务端上传文件的功能 当启动服务端后,运行客户端程序,系统提示客户在客户端输入上传文件的完整路径.当客户在客户端输入完成后,服务端实现文件上传 实现思 ...
- 【工具】Java转换exe
一.导出jar包 eclipse中对着要转换的项目,右键,导出 搜索jar,选择jar文件,下一步 选择要输出的项目 继续下一步 选择主程序 完成 二.下载及安装exe4j,并转换jar文件为exe文 ...
- Ditto剪贴板增强工具
1.简介 Ditto是一款强大的Windows剪贴板增强工具,它支持64位操作系统,而且完全免费,绿色开源,支持中文,而且还有免安装的绿色版本. 开启Ditto后,不会有任何程序界面出现,它只是默默地 ...
- .net core集成使用EasyNetQ来使用rabbitmq
之前有写到一篇介绍EasyNetQ的博文(C# .net 使用rabbitmq消息队列--EasyNetQ插件介绍),所以本文从.net core的角度去继承使用EasyNetQ,而用法类似于之前集成 ...
- C语言 Make命令教程
make只是一个根据指定的Shell命令进行构建的工具 在linux和unix中,有一个强大的使用程序,叫make,可以用它来管理多模块程序的编译和链接,直至生成可执行文件 make使用程序读取一个说 ...
- quasar框架在store中使用router跳转页面报错
网上一通百度,终于在这篇博客中找到原因. https://www.cnblogs.com/remly/p/12995936.html 原因是: 在router中导出了一个工厂函数, 既然是一个函数, ...
- reduce/reduceRight
使用 reduce 和 reduceRight 方法可以汇总数组元素的值,具体用法如下: reduce function appendCurrent (previousValue, currentVa ...