P1721 [NOI2016] 国王饮水记 题解
蒟蒻的第一篇黑题题解,求过。
题意描述
这道题用简洁的话来说,就是:
给你 \(n\) 个数字,你可以让取其中任意若干个数字,每次操作,都会使所有取的数字变为取的数字的平均数,并且你最多只能进行 \(k\) 次操作,你要在这经过最多 \(k\) 次操作后使得给你的第一个数字变得最大。输出保留 \(p\) 位的第一个数字的最后状态。
贪心策略(非正解,有助后面做题)
我们暂且不研究这道题的正解是什么,我们先看看怎么样贪心能够使得第一个数字变得最大。
首先,比第一个数字小的我们一定不需要,它们会使第一个数字变得更低
第二,我们需要知道求平均数的顺序,我们设三个水量, \(h1,h2,h3(h1 <h2<h3)\) 。现在我们通过比较不同的顺序,看如何让 \(h1\) 变得最大。
如图所示,已经是所有的顺序了:
- \(h1,h2,h3\) 同时求平均数。
- \(h1\) 先和 \(h3\) 求平均数,然后和 \(h2\) 求平均数。
- \(h1\) 先和 \(h2\) 求平均数,然后和 \(h3\) 求平均数。
其实从图中就可以看出来,第三种方案最优。
我们再用数学的方法来比较这三个方案的优劣性。
第一种方案: \(h1=(h1+h2+h3)/3\)。
第二种方案: \(h1=((h1+h3)/2+h2)/2=(h1+h3)/4+h2/2\)。
第三种方案: \(h1=((h1+h2)/2+h3)/2=(h1+h2)/4+h3/2\)。
我们通过相减比较 \(h1\) 最后的大小:
- 第一种方案减去第三种方案:
\]
\]
\]
\]
\]
第三种方案晋级。
- 第二种方案减去第三种方案:
\]
\]
\]
\]
第三种方案胜出。
总结策略
只考虑比第一个数大的数。
在一般情况下,一个一个操作(可能会超过 \(k\) 次,所以是在一般情况下)。
在一个一个操作时,从小到大操作。
斜率优化dp(正解)
由于可进行的操作次数是一定的,所以我们最后要解决的就是该在那一段区间进行操作。
首先将大于第一个数的数记录下来,排序并求出前缀和。
然后设计dp转移方程:
\(sum_i\) 为前 \(i\) 个数的前缀和(操作是赋值为平均数,并不会改变总数值,所以 \(sum_i\) 一直适用) 。
\(f_{i,j}\) 表示前面 \(i\) 个数操作了 \(j\) 次时第一个数的最大值。
得转移方程:
\]
code
for(int j=1;j<=k;j++)
for(int i=1;i<=n;i++)
for(int p=1;p<i;p++)
f[i][j]=max(f[i][j],(f[p][j-1]+sum[i]-sum[p-1])/(i-p+1));
但是这样的话,转移的时间复杂度为 \(O(n^2kp)\) ,(\(n\) 是数字个数, \(k\) 是操作次数,p为保留的精度)。
所以光是dp是不行的,我们要考虑优化dp。
我们观察这个状态转移方程:
\]
\]
由于 \(i\) 和 \(j\) 是循环中给出的,所以我们将 \(i\) 和 \(j\) 提出:
\]
经过前后对比,我们发现,设当前点为 \((i,sum_i)\) ,转移点为 \((p-1,sum_{p-1}-f_p)\) ,其实就是斜率公式。
用斜率优化dp,还有很重要的条件是要有单调性。如图,这道题目的单调性也十分的明显,由于 \(1\le h_i \le 10^5\) ,也就意味着前缀和是单调上升的。
图画的不好,谅解。
那么我们要维护的就是图中的这一个凸包。
接下来我们考虑弹出队列的数会不会对后面的操作有影响:
设当前位置为 \(i\) ,对于 \(i\) ,我们设 \(k_2\) 优于 \(k_1\) , 在这个情况下,我们可以得出:
\]
如果将 \(i+1\) 带入,结果为:
\]
比较两个式子,我们会发现,在 \(i+1\) 的情况下,如同 \(i\) 的情况相同, 原本优于 \(k_1\) 的 \(k_2\) 依然会比 \(k_2\) 更优,所以弹掉 \(k_1\) 并不会影响后面的操作。
code
for(int j=1;j<=k;j++)
{
int head=tail=1;
q[tail]=(node){1,sum[1]};
for(int i=2;i<=n;i++)
{
node x=(node){i,sum[i]};
while(head<tail&&slope(x,q[head])<slope(x,q[head+1]))head++;
f[i][j]=(f[q[head]][j-1]+sum[i]-sum[q[head]-1])/(i-q[head]+1);
while(head<tail&&slope(q[tail],q[tail-1])>slope(q[tail],i)) tail--;
q[++tail]=i;
}
}
此时我们已经将时间复杂度调小至 \(O(nkp)\) 。
\(n\le 8000\ \ k\le10^9\ \ p\le 3000\) 。
显然,我们还是不能过这道题目。
于是我们再次重审题目,看一下还有什么条件我们并没有用到。
Two thousands years later......
哦,我发,发现就怪了。
于是我参考了巨佬ljh2000的博客,终于发现了:\(h_i\) 互不相同。 那么那么就意味着......
好吧......
于是我又一次参考了巨佬ljh2000的博客,里面有两条十分重要的性质:
每一次操作的区间长度一定不比上一次操作的区间长度长!
在所有水量高度互不相同的情况下,长度大于1的区间仅有 \(O(\log{ \frac{nh}{H}})\) 个,其中 \(H=\min(h_i-h_{i-1})\) 。
首先我们证明第一个,十分简单。我们的目标是将 \(h_1\) 变得最大,而我们在一开始的时候对 \(h_i\) 进行了排序,所以越往后面, \(h_i\) 越大,越大的饼自然越要少与人分享,所以我们就会得出第一个结论。
至于第二个结论,我们只好根据众多大佬的指引,来到这里,深造一番。
Three thousands years later......
很好,没有任何的作用,看来只能等待某位大佬的指点,或者等本蒟蒻再深造几年再回答这个问题吧。
蒟蒻拙见,大佬勿喷。
P1721 [NOI2016] 国王饮水记 题解的更多相关文章
- luogu P1721 [NOI2016]国王饮水记 斜率优化dp 贪心 决策单调性
LINK:国王饮水记 看起来很不可做的样子. 但实际上还是需要先考虑贪心. 当k==1的时候 只有一次操作机会.显然可以把那些比第一个位置小的都给扔掉. 然后可以得知剩下序列中的最大值一定会被选择. ...
- uoj233/BZOJ4654/洛谷P1721 [Noi2016]国王饮水记 【dp + 斜率优化】
题目链接 uoj233 题解 下面不加证明地给出几个性质: 小于\(h[1]\)的城市一定是没用的 任何城市联通包含\(1\)且只和\(1\)联通一次 联通顺序从小到大最优 单个联通比多个一起联通要优 ...
- [UOJ#223][BZOJ4654][Noi2016]国王饮水记
[UOJ#223][BZOJ4654][Noi2016]国王饮水记 试题描述 跳蚤国有 n 个城市,伟大的跳蚤国王居住在跳蚤国首都中,即 1 号城市中.跳蚤国最大的问题就是饮水问题,由于首都中居住的跳 ...
- [Noi2016]国王饮水记
来自FallDream的博客,未经允许,请勿转载,谢谢. 跳蚤国有 n 个城市,伟大的跳蚤国王居住在跳蚤国首都中,即 1 号城市中.跳蚤国最大的问题就是饮水问题,由于首都中居住的跳蚤实在太多,跳蚤国王 ...
- BZOJ4654/UOJ223 [Noi2016]国王饮水记
本文版权归ljh2000和博客园共有,欢迎转载,但须保留此声明,并给出原文链接,谢谢合作. 本文作者:ljh2000 作者博客:http://www.cnblogs.com/ljh2000-jump/ ...
- BZOJ4654 NOI2016国王饮水记(动态规划+三分)
有很多比较显然的性质.首先每个城市(除1外)至多被连通一次,否则没有意义.其次将城市按水位从大到小排序后,用以连通的城市集合是一段前缀,并且不应存在比1城市还小的.然后如果确定了选取的城市集合,每次选 ...
- 【BZOJ4654】【NOI2016】国王饮水记(动态规划,斜率优化)
[BZOJ4654][NOI2016]国王饮水记(动态规划,斜率优化) 题面 BZOJ 洛谷 题解 首先肯定是找性质. 明确一点,比\(h_1\)小的没有任何意义. 所以我们按照\(h\)排序,那么\ ...
- *UOJ#223. 【NOI2016】国王饮水记
$n \leq 8000$的数列,问不超过$m \leq 1e9$次操作后第一个数字最大是多少.操作:选一些数,把他们变成他们的平均值.需要保留$p \leq 3000$位小数,提供了一个小数高精度库 ...
- [NOI 2016]国王饮水记
Description 题库链接 给出 \(n\) 个水杯,每个水杯装有不同高度的水 \(h_i\) ,每次可以指定任意多水杯用连通器连通后断开,问不超过 \(k\) 次操作之后 \(1\) 号水杯的 ...
随机推荐
- AppWeb认证绕过漏洞(CVE-2018-8715)
影响范围 Appweb 7.0.2及早期版本. 复现 构造头Authorization: Digest username=admin 返回包里包含session 发送POST请求,添加session到 ...
- BUU八月份水题记录
目录 [BJDCTF 2nd]fake google(SSTI) [BJDCTF2020]Easy MD5(md5注入) [ZJCTF 2019]NiZhuanSiWei(反序列化) [BJDCTF ...
- C++ //多继承语法 C++中允许一个类继承多个类
1 //多继承语法 C++中允许一个类继承多个类 2 #include <iostream> 3 #include <string> 4 using namespace std ...
- Flutter 中的动画
Flutter 中动画的创建有很多种, 需要根据具体的需求选择不同的动画.如果只是简单的布局等的动画直接使用最简单的隐式动画就可以了,因为隐式动画是由框架控制的,所以仅仅只需要更改变需要变化属性就可以 ...
- 关于表单重复提交之验证码 和谷歌Kaptcha图片验证码的使用
表单重复提交之-----验证码 表单重复提交有三种常见的情况: 一:提交完表单.服务器使用请求转来进行页面跳转.这个时候,用户按下功能键 F5,就会发起最后一次的请求. 造成表单重复提交问题.解决方法 ...
- 跟我一起写 Makefile(九)
使用函数 ---- 在Makefile中可以使用函数来处理变量,从而让我们的命令或是规则更为的灵活和具有智能.make所支持的函数也不算很多,不过已经足够我们的操作了.函数调用后,函数的返回值可以当做 ...
- Notes about "Exploring Expect"
Chapter 3 Section "The expect Command": expect_out(0,string) can NOT be written as "e ...
- CMMI相关图书
Integrating CMMI and Agile Development: Case Studies and Proven Techniques for Faster Performance Im ...
- MySQL高级查询 & 事务机制
1.基础查询where 高级条件查询Where子句 SELECT empno,ename,sal,hiredate FROM t_tmp WHERE deptno=10 AND (sal+IFNULL ...
- Python - typing 模块 —— Any Type
前言 typing 是在 python 3.5 才有的模块 前置学习 Python 类型提示:https://www.cnblogs.com/poloyy/p/15145380.html 常用类型提示 ...