[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 从一个 ...
随机推荐
- Centos6.5安装rar5.3
linux下使用最多的压缩工具是gzip,zip等,如果需要使用rar,就必须编译安装了,以下是编译安装rar教程: 一.安装支持库yum install -y gcc gcc-c++ autocon ...
- 【JUC】JDK1.8源码分析之ConcurrentHashMap
一.前言 最近几天忙着做点别的东西,今天终于有时间分析源码了,看源码感觉很爽,并且发现ConcurrentHashMap在JDK1.8版本与之前的版本在并发控制上存在很大的差别,很有必要进行认真的分析 ...
- -bash: make: command not found
yum -y install gcc automake autoconf libtool make
- springboot EL @Value
一,springboot中 看一下代码: @Controller public class HelloController { //读取枚举值 @Value("#{T(com.example ...
- Linux CentOS 6.7 挂载U盘
1. 首先查看U盘是否成功安装fdisk -l 2. 在/mnt下创建U盘目录mkdir /mnt/usb 3. 挂载U盘mount -t vfat /dev/sdb1 /mnt/usb 4. 卸载U ...
- luoguP1154 奶牛分厩 [数论]
题目描述 农夫约翰有N(1<=N<=5000)头奶牛,每头奶牛都有一个唯一的不同于其它奶牛的编号Si,所有的奶牛都睡在一个有K个厩的谷仓中,厩的编号为0到K-1.每头奶牛都知道自己该睡在哪 ...
- Eclipse中如何使用Hibernate
首先创建一个java web项目,其目录如下: (1)创建文件夹hibernate4(用于存放下载的hibernate工具包lib/required文件夹下所有的jar包),jar包目录如下: (2) ...
- LUOGU P4253 [SCOI2015]小凸玩密室(树形dp)
传送门 解题思路 玄学树形\(dp\),题目描述极其混乱...看错了两次题,设首先根据每次必须点完子树里的灯才能点别的,那么点灯情况只有两种,第一种是点到某一个祖先,第二种是点到某一个祖先的兄弟.所以 ...
- SpringBoot_03_SpringBoot对其他技术的整合
1.SpringBoot整合Mybatis 1.2 添加Mybatis的起步依赖 <!--mybatis起步依赖--> <dependency> <groupId> ...
- csdn的富文本编辑器(html)格式的文章,添加的代码格式粘贴到项目中总是不对,我这里是maven的pom文件。
在用富文本编辑器(html)格式编辑的时候,总是出现将pom文件的配置从csdn粘贴到项目中发现很多红,是因为自己在设置的时候有问题. 因为我们的是pom.xml,这里原来设置的java呀或者js为了 ...