poj2018 Best Cow Fences[二分答案or凸包优化]
首先暴力很好搞,但是优化的话就不会了。放弃QWQ。
做法1:二分答案
然后发现平均值是$ave=\frac{sum}{len}$,这种形式似乎可以二分答案?把$len$移到左边。
于是二分$ave$,去找数列有没有区间和大于等于其$len$乘以$ave$的,然后卡住了。。
有一个很巧的转化,把每个数都减去一个$ave$,然后任意区间和就相当于把$ave$累加了$len$次。
于是乎只要看区间和$S-len*ave$是否大于等于0就可以了。
存在这样一个区间就说明$ave$可以更大,否则要小。查找存在性是一个简单的前缀和dp。
$O(nlogn)$。
poj精度死活卡不过去。
看来技巧还不够。
于是一气之下把所有double型全换成long long,也就是把小数点后五位通通看成整数,然后运算,最后除以100。
唉。
启示:关于平均数的题目很多时候都是用二分答案处理的。因为是分数形式。
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#define dbg(x) cerr << #x << " = " << x <<endl
using namespace std;
typedef long long ll;
typedef double db;
typedef pair<int,int> pii;
template<typename T>inline T _min(T A,T B){return A<B?A:B;}
template<typename T>inline T _max(T A,T B){return A>B?A:B;}
template<typename T>inline char MIN(T&A,T B){return A>B?(A=B,):;}
template<typename T>inline char MAX(T&A,T B){return A<B?(A=B,):;}
template<typename T>inline void _swap(T&A,T&B){A^=B^=A^=B;}
template<typename T>inline T read(T&x){
x=;int f=;char c;while(!isdigit(c=getchar()))if(c=='-')f=;
while(isdigit(c))x=x*+(c&),c=getchar();return f?x=-x:x;
}
const int N=1e5+;
const db eps=1e-;
ll A[N],s[N];
ll L=,R,mid;
int n,l;
inline bool check(ll ave){
for(register int i=;i<l;++i)s[i]=s[i-]+A[i]-ave;
ll minx=1e16;
for(register int i=l;i<=n;++i){
MIN(minx,s[i-l]);s[i]=s[i-]+A[i]-ave;
if(s[i]-minx>=)return ;
}
return ;
} int main(){//freopen("test.in","r",stdin);//freopen("test.out","w",stdout);
read(n),read(l);
for(register int i=;i<=n;++i)A[i]=read(A[i])*,MAX(R,A[i]);
while(L<R){
mid=L+R+>>;
if(check(mid))L=mid;
else R=mid-;
}
printf("%lld",(L/));
return ;
}
做法2:凸包优化
发现以每个点结尾的平均值$ave_i=\frac{S_i-S_j}{i-j}$ $0 \leq j \leq m$
于是想到斜率。问题变成:在前面的区间内找一点与$i$号点连线斜率最大。
然后发现上凸的3个点,中间那个没有用,所以维护一个下凸包。
查找的话是需要二分的,二分找下凸包里斜率$k_{i-1}<k_{i}>k_{i+1}$的这个点。
但是,基于这题并不是让求每个点结尾的最大平均数,而是全局的。
所以有这么一个神仙思路优化复杂度:
在查找时不要二分了,从队首开始比较,若$k_{l,i}<k_{l+1,i}$则弹出队首。
一直到找到这个最大斜率点,作为这个$i$的局部答案。
如果之后出现一个点,他和之前弹出过的点的斜率比目前没有弹出的点间的斜率都要大呢?
是有这种情况的。

但是,他不会影响全局最优解的形成,即使对于以$i$结尾算错了也没关系。
为什么呢。观察上面这种情况,在1点把下凸包前面一堆全弹掉了。发生这种情况当且仅当这个1号点在橙色线上方。
而2号点那种斜率最大的出现在已经弹出了的点里,必须满足在橙色线下方。
一旦有弹出,那么1号点决策一定比2号点的优。所以2号点算错了也不影响全局最大值。
所以这样的做法是$O(n)$获取全局最优解的。
所以网上说的都是并不严谨的。
这个$O(n)$做法正确性的证明是由hkk神仙提出来的,感谢!大家觉得好可以多去膜他。
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#define dbg(x) cerr << #x << " = " << x <<endl
using namespace std;
typedef long long ll;
typedef double db;
typedef pair<int,int> pii;
template<typename T>inline T _min(T A,T B){return A<B?A:B;}
template<typename T>inline T _max(T A,T B){return A>B?A:B;}
template<typename T>inline char MIN(T&A,T B){return A>B?(A=B,):;}
template<typename T>inline char MAX(T&A,T B){return A<B?(A=B,):;}
template<typename T>inline void _swap(T&A,T&B){A^=B^=A^=B;}
template<typename T>inline T read(T&x){
x=;int f=;char c;while(!isdigit(c=getchar()))if(c=='-')f=;
while(isdigit(c))x=x*+(c&),c=getchar();return f?x=-x:x;
}
const int N=1e5+;
db ans;
int n,m;
int S[N],q[N],l=,r; int main(){//freopen("test.in","r",stdin);//freopen("test.out","w",stdout);
read(n),read(m);
for(register int i=;i<=n;++i)S[i]=S[i-]+read(S[i]);
for(register int i=m;i<=n;++i){
while(l<r&&(S[i-m]-S[q[r]])*1ll*(q[r]-q[r-])<=(S[q[r]]-S[q[r-]])*1ll*(i-m-q[r]))--r;
q[++r]=i-m;
while(l<r&&(S[i]-S[q[l]])*1ll*(i-q[l+])<=(S[i]-S[q[l+]])*1ll*(i-q[l]))++l;
MAX(ans,(db)(S[i]-S[q[l]])/(db)(i-q[l]));
}
printf("%d\n",(int)(ans*));
return ;
}
poj2018 Best Cow Fences[二分答案or凸包优化]的更多相关文章
- POJ-2018 Best Cow Fences(二分加DP)
Best Cow Fences Time Limit: 1000MS Memory Limit: 30000K Total Submissions: 10174 Accepted: 3294 Desc ...
- loj#10012\poj2018 Best Cow Fences(二分)
题目 #10012 「一本通 1.2 例 2」Best Cow Fences 解析 有序列\(\{a_i\}\),设\([l,r]\)上的平均值为\(\bar{x}\),有\(\sum_{i=l}^r ...
- POJ-2018 Best Cow Fences 二分
题意:找到一个连续区间,区间的长度至少大于f,现在要求这个区间的平均值最大. 题解: 二分找答案. 每次对于2分的mid值, 都把原来的区间减去mid, 然后找到一长度至少为f的区间, 他们的区间和& ...
- POJ2018 Best Cow Fences 二分
实数折磨人啊啊啊啊啊啊啊 好,实数应该是最反人类的东西了...... 这个害得我调了0.5天才过. 大意是这样的:给你一个数列,求其中不少于f个的连续数的最大平均值. 不禁想起寒假的课程来... 此处 ...
- POJ2018 Best Cow Fences —— 斜率优化DP
题目链接:https://vjudge.net/problem/POJ-2018 Best Cow Fences Time Limit: 1000MS Memory Limit: 30000K T ...
- bzoj 2402: 陶陶的难题II 二分答案维护凸包
2402: 陶陶的难题II Time Limit: 40 Sec Memory Limit: 128 MBSec Special JudgeSubmit: 68 Solved: 45[Submi ...
- Poj2018 Best Cow Fences
传送门 题目大意就是给定一个长度为 n 的正整数序列 A ,求一个平均数最大的,长度不小于 L 的子序列. 思路: 二分答案. Code: #include<iostream> #incl ...
- poj2018——Best Cow Fences
Description Farmer John's farm consists of a long row of N (1 <= N <= 100,000)fields. Each fie ...
- Poj 2018 Best Cow Fences(分数规划+DP&&斜率优化)
Best Cow Fences Time Limit: 1000MS Memory Limit: 30000K Description Farmer John's farm consists of a ...
随机推荐
- CornerNet 算法笔记
论文名称:CornerNet: Detecting Objects as Paired Keypoints 论文链接:https://arxiv.org/abs/1808.01244 代码链接:htt ...
- 自然语言处理NLP学习笔记二:NLP实战-开源工具tensorflow与jiagu使用
前言: NLP工具有人推荐使用spacy,有人推荐使用tensorflow. tensorflow:中文译作:张量(超过3维的叫张量)详细资料参考:http://www.tensorfly.cn/ J ...
- 论文翻译:LP-3DCNN: Unveiling Local Phase in 3D Convolutional Neural Networks
引言 传统的3D卷积神经网络(CNN)计算成本高,内存密集,容易过度拟合,最重要的是,需要改进其特征学习能力.为了解决这些问题,我们提出了整流局部相位体积(ReLPV)模块,它是标准3D卷积层的有效替 ...
- 一篇文章搞懂python2、3编码
说在前边: 编码问题一直困扰着每一个程序员的编程之路,如果不将它彻底搞清楚,那么你的的这条路一定会走的格外艰辛,尤其是针对使用python的程序员来说,这一问题更加显著, 因为python有两个版本, ...
- shell提升篇
6. 条件判断 1.基本语法 [ condition ](注意condition前后要有空格) 注意:条件非空即为true,[ fsdm ]返回true,[] 返回false. 2. 常用判断条件 ( ...
- 01串LIS(固定串思维)--Kirk and a Binary String (hard version)---Codeforces Round #581 (Div. 2)
题意:https://codeforc.es/problemset/problem/1204/D2 给你一个01串,如:0111001100111011101000,让你改这个串(使0尽可能多,任意 ...
- 基于VS搭建OpenCV环境
OpenCV OpenCV的全称是Open Source Computer Vision Library,是一个跨平台的计算机视觉库.OpenCV是由英特尔公司发起并参与开发,以BSD许可证授权发行, ...
- Docker 构建私有镜像仓库
在使用Docker一段时间后,往往会发现手头积累了大量的自定义镜像文件,这些文件通过公有仓库进行管理并不方便,另外有时候只是希望在内部用户之间进行分享,不希望暴露出去.这种情况下,就有必要搭建一个本地 ...
- sed---流文本操作
一:sed基本命令 sed的使用格式 sed [optiona] 'command' files sed 参数[-nefir] 动作[n1,[n2]] function sed -n:只有经过sed特 ...
- MySQL性能优化(二):优化数据库的设计
原文:MySQL性能优化(二):优化数据库的设计 版权声明:本文为博主原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明. 本文链接:https://blog.csdn.n ...