题解 P1721 【[NOI2016]国王饮水记】
洛谷的
首先肯定是找性质。
明确一点,比
当次数不够的时候,我们肯定不能只合并一个位置,显然合并所有位置还是更优的。
那么,既然不能够一个个合并,所以只能够把若干次合并放在一起做。
因为每次和后面的若干个做完合并之后,这些一起合并的位置就可以直接丢掉了,
再因为从小往大合并更优,假设
这个式子很像斜率:
相当于把前面所有已知状态看成一个点
现在可以做到我不会我不会,可以参考这里的证明
那么这样子,只需要
完整的代码戳这里
以下是阉割版本。毕竟放个700行的代码翻都翻不完
#include<iostream>
#include<cstdio>
#include<cstring>
#include<string>
#include<cstdlib>
#include<algorithm>
using namespace std;
const int MAX=8080;
Decimal ans;
int n,K,p,h[MAX],zy[MAX][15],s[MAX],tot;
int Q[MAX],H,T;
double f[MAX][15];
struct Node{double x,y;}q[MAX];
double Slope(Node a,Node b){return (a.y-b.y)/(a.x-b.x);}
Decimal Calc(int i,int j)
{
if(!j)return h[1];
return (Calc(zy[i][j],j-1)+s[i]-s[zy[i][j]])/(i-zy[i][j]+1);
}
int main()
{
n=read();K=read();p=read();h[tot=1]=read();
for(int i=2;i<=n;++i)
{
h[i]=read();
if(h[i]>h[1])h[++tot]=h[i];
}
n=tot;sort(&h[1],&h[n+1]);
for(int i=1;i<=n;++i)s[i]=s[i-1]+h[i];
K=min(K,n);
for(int i=1;i<=n;++i)f[i][0]=h[1];
int lim=min(K,14);
for(int j=1;j<=lim;++j)
{
Q[H=T=1]=1;
for(int i=1;i<=n;++i)q[i]=(Node){i-1,s[i]-f[i][j-1]};
for(int i=2;i<=n;++i)
{
Node u=(Node){i,s[i]};
while(H<T&&Slope(u,q[Q[H]])<Slope(u,q[Q[H+1]]))++H;
zy[i][j]=Q[H];f[i][j]=(s[i]-s[Q[H]]+f[Q[H]][j-1])/(i-Q[H]+1);
while(H<T&&Slope(q[Q[T]],q[Q[T-1]])>Slope(q[Q[T]],q[i]))--T;
Q[++T]=i;
}
}
int m=n-K+lim,pos;double mx=0;
for(int j=0;j<=lim;++j)
if(f[m][j]>mx)mx=f[m][j],pos=j;
ans=Calc(m,pos);
for(int i=m+1;i<=n;++i)ans=(ans+h[i])/2;
cout<<ans.to_string(p<<1)<<endl;
return 0;
}