[JZOJ4640] 【GDOI2017模拟7.15】妖怪
题目
描述
题目大意
给你一堆aia_iai和bib_ibi(方便起见用的变量和上面不一样),让你搞出一个xxx(相当于题目中的ba\frac{b}{a}ab,随便推推就能知道),
使得maxai+bi+aix+bix\max a_i+b_i+a_ix+\frac{b_i}{x}maxai+bi+aix+xbi最小。
思考历程
第一眼看下去,最大最小放一起,显然就是一个二分啊!
然后开始想……想不出来,推了个式子,感觉似乎要三分套三分套三分……
更气的是这题还不好打暴力。
所以推了很久之后什么都没有打。
正解
其实这个题目的正解有好几种。
先说二分的做法(WMY大佬的方法,只可惜被卡了常数):
首先我们二分答案,然后判断是否可行。
要满足ai+bi+aix+bix≤ansa_i+b_i+a_ix+\frac{b_i}{x}\leq ansai+bi+aix+xbi≤ans,
变化一下式子:aix2+(ai+bi−ans)x+bi≤0a_ix^2+(a_i+b_i-ans)x+b_i\leq 0aix2+(ai+bi−ans)x+bi≤0
发现可以用一元二次方程的方法来解,
于是对于每个aia_iai和bib_ibi,我们都可以得出一个解集。
然后取它们的交集,如果不为空就成立。
正确性显然。
再说三分的做法:
首先有个重要的结论:y=ai+bi+aix+bixy=a_i+b_i+a_ix+\frac{b_i}{x}y=ai+bi+aix+xbi的图像是一个单峰函数(VVV字形,左边陡,右边缓)。
LYL给出了一个很强的证明:
首先ai+bia_i+b_iai+bi是定值,先不理它,只考虑y=aix+bixy=a_ix+\frac{b_i}{x}y=aix+xbi
变化式子:aix2−yx+bi=0a_ix^2-yx+b_i=0aix2−yx+bi=0
一元二次方程!然后算出Δ=y2−4aibi\Delta=y^2-4a_ib_iΔ=y2−4aibi,解为x=y±y2−4aibi2ax=\frac{y \pm\sqrt{y^2-4a_ib_i}}{2a}x=2ay±y2−4aibi
这时候就可以脑补出它的图像了……
显然,方程的解只有一个的时候就是顶点,所以顶点为(aba,2ab)(\frac{\sqrt{ab}}{a},2\sqrt{ab})(aab,2ab)
还有YMQ的证明:
将yyy除以aia_iai,设ci=biaic_i=\frac{b_i}{a_i}ci=aibi,则y=x+cixy=x+\frac{c_i}{x}y=x+xci
这就相当于反比例函数上的纵坐标和横坐标之和!
显然在直线y=xy=xy=x上最优……(具体证明可以用基本不等式)
整理一下,当x=abax=\frac{\sqrt{ab}}{a}x=aab时最优。
证明完了,下面是做法:
对于每个iii,都会有一个图象。将它们放在一起,取maxmaxmax,可以发现图象是单峰(谷)的。
考虑反证,如果图象为WWW形,那么中间交接的那个地方实际上可以继续延伸,在上面更高的地方形成VVV形,所以不可能会出现WWW形。
所以三分出最低点就可以了。
还有一种方法是最优秀的线性方法。
考虑斜率优化。
假设ai<aja_i<a_jai<aj且yi<yjy_i<y_jyi<yj,那么ai+bi+aix+bix<aj+bj+ajx+bjxa_i+b_i+a_ix+\frac{b_i}{x}<a_j+b_j+a_jx+\frac{b_j}{x}ai+bi+aix+xbi<aj+bj+ajx+xbj
变化式子:T(i,j)=−bi−bjai−aj<xT(i,j)=-\frac{b_i-b_j}{a_i-a_j}<xT(i,j)=−ai−ajbi−bj<x(TTT只是为了后面方便表示)
然后就可以斜率优化了!
先处理出一个斜率递增的序列,对于序列上的每个点iii,在x∈[T(i−1,i),T(i,i+1)]x\in [T(i-1,i),T(i,i+1)]x∈[T(i−1,i),T(i,i+1)]时,yiy_iyi是最大的。
所以求出xxx在这个区间内yiy_iyi的最小值就好了。
将yiy_iyi的VVV形图象画出来,就可以发现这个区间的位置及对应的最小值的情况。
在顶点左边,跨过顶点,在顶点右边三种情况分类讨论。
这就可以O(1)O(1)O(1)求出它的最小值。
总的来说,这个方法是线性的。
代码
线性做法(我只打了线性的)
using namespace std;
#include <cstdio>
#include <cstring>
#include <cmath>
#include <algorithm>
inline int input(){
char ch=getchar();
while (ch<'0' || '9'<ch)
ch=getchar();
int x=0;
do{
x=x*10+ch-'0';
ch=getchar();
}
while ('0'<=ch && ch<='9');
return x;
}
#define N 1000010
int n;
struct Monster{
int a,b;
} d[N];
inline bool cmpd(const Monster &x,const Monster &y){
return x.a<y.a || x.a==y.a && x.b>y.b;
}
int q[N],head,tail;
inline bool pd(int i,int j,int k){
return -(long long)(d[i].b-d[j].b)*(d[j].a-d[k].a)>=-(long long)(d[j].b-d[k].b)*(d[i].a-d[j].a);
}
inline double calc(int i,int j){
return -(double)(d[i].b-d[j].b)/(d[i].a-d[j].a);
}
inline double get(int k,double l,double r){
if (l-r>1e-8)
return 1e8;
double mn=sqrt(1ll*d[k].a*d[k].b)/d[k].a;
if (r<mn)
return d[k].a+d[k].b+d[k].a*r+d[k].b/r;
if (l>mn)
return d[k].a+d[k].b+d[k].a*l+d[k].b/l;
return d[k].a+d[k].b+d[k].a*mn+d[k].b/mn;
}
int main(){
n=input();
for (int i=1;i<=n;++i)
d[i]={input(),input()};
sort(d+1,d+n+1,cmpd);
for (int i=1;i<=n;++i){
while (head<tail && pd(q[tail-1],q[tail],i))
tail--;
q[++tail]=i;
}
double ans=get(q[1],1e-4,calc(q[1],q[2]));
for (int i=2;i<tail;++i)
ans=min(ans,get(q[i],calc(q[i-1],q[i]),calc(q[i],q[i+1])));
ans=min(ans,get(q[tail],calc(q[tail-1],q[tail]),1e8));
printf("%.4lf",ans);
return 0;
}
自认为讲解得比较清晰,就不打注释了。
总结
面对这样有关式子和最值的题目,二分和三分都是很好的思考方向。
有时候还可以尝试一下斜率优化。
最后我们认识了这样的函数y=ax+bxy=ax+\frac{b}{x}y=ax+xb,它的顶点在(aba,2ab)(\frac{\sqrt{ab}}{a},2\sqrt{ab})(aab,2ab)
[JZOJ4640] 【GDOI2017模拟7.15】妖怪的更多相关文章
- [JZOJ4633] 【GDOI2017模拟7.15】萌萌哒
题目 描述 题目大意 给你一个数列,接下来有许多个操作,使得区间[l1,r1][l_1,r_1][l1,r1]和[l2,r2][l_2,r_2][l2,r2]对应的位置染上同样的颜色(使得它们 ...
- 【GDOI2016模拟3.15】基因合成(回文串+性质+DP)
[GDOI2016模拟3.15]基因合成 题意: 给一个目标串,要求从空串进行最少的操作次数变成目标串,操作有两种: 在串的头或尾加入一个字符. 把串复制一遍后反向接到串的末尾. 因为有回文操作,所以 ...
- NOIP2017提高组 模拟赛15(总结)
NOIP2017提高组 模拟赛15(总结) 第一题 讨厌整除的小明 [题目描述] 小明作为一个数学迷,总会出于数字的一些性质喜欢上某个数字,然而当他喜欢数字k的时候,却十分讨厌那些能够整除k而比k小的 ...
- NOIP模拟赛15
NOIP2017金秋冲刺训练营杯联赛模拟大奖赛第一轮Day1 T1 天天去哪儿吃 直接枚举 #include<cstdio> #include<algorithm> using ...
- JZOJ.5281【NOIP2017模拟8.15】钦点
Description
- noip模拟赛#15
#15 T1:a[i]>=a[i/2].输出a的最大字典序 =>可以发现这是二叉树的情况那么就先预处理出每个点有多少个儿子然后递归处理就可以了. #include<cstdio> ...
- [考试反思]0809NOIP模拟测试15:解剖
说在前面: 不建议阅读.这里没有考试经验,只有一大堆负面情绪. 看了你不会有什么收获.看完了就不要怪我影响了你的心情. 以后不粘排行榜了.没什么意思没什么用. 但是我的意思并不是因为这次没考好的一时兴 ...
- [NOIP2018模拟10.15]比赛报告
闲扯 昨晚又颓到好晚,Yali的降智光环感觉持续至今... 题面好评 T1T3都玩过 逃) T1没看多久就开始写二分+并查集 然后T3看着眼熟想了一个多小时...结果啥都没想出来 赶紧看T2发现还是没 ...
- noi.ac NOI挑战营模拟赛1-5
注:因为博主是个每次考试都爆零垫底的菜鸡,所以此篇博客很有可能咕咕咕 (指只贴AC代码不写题解的......如果我真的不会做的话,就不能怪我了qwqwq) Day1 T1 swap 23pts 从一个 ...
随机推荐
- 使用Docker创建数据容器
使用Docker创建数据容器 翻译自: Data-only container madness 1.什么是数据容器? 数据容器就是本身只创建一个volume供其他容器共享,创建完后即退出,不执行任何任 ...
- Qt无边框窗口的移动、拉伸边框、鼠标滚轮缩放大小
主要是处理窗口上鼠标的几个事件,具体代码请看下面的截图, 完整代码的下载链接在此:http://download.csdn.net/detail/beyond0824/9657110, 本示例代码中, ...
- Spring Boot集成Shiro实战
Spring Boot集成Shiro权限验证框架,可参考: https://shiro.apache.org/spring-boot.html 引入依赖 <dependency> < ...
- pathlib生成文件的软链接
在训练深度网络时,保存模型,想要维护一个latest.t7的文件,想到给最好的模型创建一个软链接到latest.t7 这样模型不占地,还能便于后续脚本加载最好模型 起初是看到mmdetection中是 ...
- Jmeter---参数化之用户参数
总结: 参数化几次就要设置几个线程,执行的时候,是按顺序执行,下面的请求也会跟着请求
- D3.js坐标轴的绘制方法、添加坐标轴的刻度和各比例尺的坐标轴(V3版本)
坐标轴(Axis) 坐标轴(Axis)在很多图表中都可见到,例如柱形图.折线图.散点图等.坐标轴由一组线段和文字组成,坐标轴上的点由一个坐标值确定.但是,如果使用SVG的直线和文字一笔一画的绘制坐 ...
- iOS逆向系列-tweak补充
tweak加载资源 开发自己的deb插件需要加载自己的资源,比如图片资源.iOS中常用的两种加载图片资源的方式: + (nullable UIImage *)imageNamed:(NSString ...
- RabbitMQ学习第二记:工作队列的两种分发方式,轮询分发(Round-robin)和 公平分发(Fair dispatch)
1.什么是RabbitMQ工作队列 我们在应用程序使用消息系统时,一般情况下生产者往队列里插入数据时速度是比较快的,但是消费者消费数据往往涉及到一些业务逻辑处理导致速度跟不上生产者生产数据.因此如果一 ...
- COGS2356 【HZOI2015】有标号的DAG计数 IV
题面 题目描述 给定一正整数n,对n个点有标号的有向无环图进行计数. 这里加一个限制:此图必须是弱连通图. 输出答案mod 998244353的结果 输入格式 一个正整数n. 输出格式 一个数,表示答 ...
- Educational Codeforces Round 27 D. Driving Test
单调栈 题意看了半天... #include <cstdio> #include <cstdlib> #include <cmath> #include <c ...