Portal -->bzoj2402

Solution

  这题的话,看到答案的形式想到分数规划(Portal -->【learning】)

  套路一波,记当前二分的\(mid\)为\(\lambda'\),那么其实就是要快速判断:

\[\begin{aligned}
&(y_i+q_j)-\lambda'(x_i+p_j)\\
=&(y_i-\lambda' x_i)+(q_j-\lambda' p_j)
\end{aligned}
\]

  的最大值与\(0\)的大小关系

  

  上面的这个式子虽然说与\(i\)和\(j\)有关,但是两个部分的形式是完全一样并且相互不影响的,可以看成一个函数\(g\),那么对于每次询问我们只要求出\(x\)到\(y\)路径上的\(g\)然后找最大值和次大值加起来就好了

  求\(x\)到\(y\)路径上的信息我们可以考虑用树链剖分,接下来的问题就是怎么求一条重链中所有点的\(g\)的最大值

​   

  观察\(g\)的形式:

\[\begin{aligned}
g(i)&=y_i-\lambda'x_i\\
y_i&=\lambda'x_i+g(i)
\end{aligned}
\]

  那其实可以转化成一个这样的问题:求斜率为\(\lambda'\)且过\((x_i,y_i)\)的直线的截距的最大值

​  容易得出结论最优的\((x_i,y_i)\)一定是在上凸壳上的

  那所以我们考虑对于树剖中的线段树的每一个区间,分别维护这个区间中的点对应的\((x_i,y_i)\)和\((q_i,p_i)\)的上凸壳,查询的话直接在上凸壳上二分就好了

  

  一些需要注意的地方:

1、在查询二分的时候,由于题目没有保证任意两个点的\(x,y,p,q\)不相同,所以不能直接用除法,而是应该移下项用乘法判断

2、额如果是用vector存凸包的话。。在求凸包的时候。。要注意下标是从\(0\)开始的!\(0\)开始的!\(0\)开始的!嗯。。

3、然后我的叉积好像写的有点丑了不过!不管了qwq

  

  代码大概长这样

#include<iostream>
#include<cstdio>
#include<cstring>
#include<vector>
#include<algorithm>
#define Pr pair<double,double>
#define mp make_pair
using namespace std;
const int N=1e5+10,SEG=N*4,inf=2147483647;
const double eps=1e-5;
struct xxx{
int y,nxt;
}a[N*2];
double val[N][4];
int h[N],top[N],son[N],sz[N],pre[N],dep[N];
int lis[N],dfn[N];
int n,m,tot,dfn_t;
void update(Pr &ret,Pr data);
void add(int x,int y);
void dfs1(int fa,int x,int d);
void dfs2(int fa,int x);
bool check(int x,int y,double k);
void solve(int x,int y); struct Hull{/*{{{*/
vector<Pr> a;
void insert(double x,double y){a.push_back(mp(x,y));}
double chaji(Pr x,Pr y){return x.first*y.second-x.second*y.first;}
void build(){
sort(a.begin(),a.end());
vector<Pr> st;
st.clear();
int top=0;
for (int i=0;i<a.size();++i){
while (top>1&&chaji(mp(a[i].first-st[top-2].first,a[i].second-st[top-2].second),mp(st[top-1].first-st[top-2].first,st[top-1].second-st[top-2].second))<0)
//!!因为是从0开始的所以st的下标都要-1
--top,st.pop_back();
++top; st.push_back(a[i]);
}
a=st;
a.resize(top);
}
double query(double k){
int l=0,r=a.size()-1,mid;
while (l<r){
mid=l+r>>1;
if (a[mid+1].second-a[mid].second<=k*(a[mid+1].first-a[mid].first)) r=mid;
else l=mid+1;
}
return a[l].second-k*a[l].first;
}
void debug(){
for (int i=0;i<a.size();++i) printf("%.2lf %.2lf\n",a[i].first,a[i].second);
}
};/*}}}*/
namespace Seg{/*{{{*/
int ch[SEG][2];
Hull info[SEG][2];
int n,tot;
void _build(int x,int l,int r){
int tmp;
for (int i=l;i<=r;++i){
tmp=lis[i];
info[x][0].insert(val[tmp][0],val[tmp][1]);
info[x][1].insert(val[tmp][2],val[tmp][3]);
}
info[x][0].build(); info[x][1].build();
if (l==r) return;
int mid=l+r>>1;
ch[x][0]=++tot; _build(ch[x][0],l,mid);
ch[x][1]=++tot; _build(ch[x][1],mid+1,r);
}
void build(int _n){n=_n; tot=1; _build(1,1,n);}
Pr _query(int x,int l,int r,int lx,int rx,double k){
if (l<=lx&&rx<=r){
return mp(info[x][0].query(k),info[x][1].query(k));
}
Pr ret=mp(-inf,-inf);
int mid=lx+rx>>1;
if (l<=mid) update(ret,_query(ch[x][0],l,r,lx,mid,k));
if (r>mid) update(ret,_query(ch[x][1],l,r,mid+1,rx,k));
return ret;
}
Pr query(int l,int r,double k){return _query(1,l,r,1,n,k);}
}/*}}}*/ int main(){
#ifndef ONLINE_JUDGE
freopen("a.in","r",stdin);
#endif
int x,y;
scanf("%d",&n);
for (int j=0;j<4;++j)
for (int i=1;i<=n;++i)
scanf("%lf",&val[i][j]);
memset(h,-1,sizeof(h));
tot=0;
for (int i=1;i<n;++i){
scanf("%d%d",&x,&y);
add(x,y); add(y,x);
}
dfs1(0,1,1);
top[1]=1; dfn_t=0;
dfs2(0,1);
Seg::build(n);
scanf("%d",&m);
for (int i=1;i<=m;++i){
scanf("%d%d",&x,&y);
solve(x,y);
}
} void add(int x,int y){
a[++tot].y=y; a[tot].nxt=h[x]; h[x]=tot;
} void dfs1(int fa,int x,int d){
int u;
son[x]=0; sz[x]=1; pre[x]=fa; dep[x]=d;
for (int i=h[x];i!=-1;i=a[i].nxt){
u=a[i].y;
if (u==fa) continue;
dfs1(x,u,d+1);
if (sz[son[x]]<sz[u]) son[x]=u;
sz[x]+=sz[u];
}
} void dfs2(int fa,int x){
int u;
dfn[x]=++dfn_t; lis[dfn[x]]=x;
if (son[x]){
top[son[x]]=top[x];
dfs2(x,son[x]);
}
for (int i=h[x];i!=-1;i=a[i].nxt){
u=a[i].y;
if (u==fa||u==son[x]) continue;
top[u]=u;
dfs2(x,u);
}
} bool check(int x,int y,double k){
Pr ret=mp(-inf,-inf);
while (top[x]!=top[y]){
if (dep[top[x]]<dep[top[y]]) swap(x,y);
update(ret,Seg::query(dfn[top[x]],dfn[x],k));
x=pre[top[x]];
}
if (dfn[x]>dfn[y]) swap(x,y);
update(ret,Seg::query(dfn[x],dfn[y],k));
return ret.first+ret.second>0;
} void update(Pr &ret,Pr data){
ret.first=max(ret.first,data.first);
ret.second=max(ret.second,data.second);
} void solve(int x,int y){
double l=0,r=1e5,mid,ans;
while (r-l>eps){
mid=(l+r)*0.5;
if (check(x,y,mid)) l=mid;
else r=mid;
}
printf("%.4lf\n",l);
}

【bzoj2402】陶陶的难题II的更多相关文章

  1. [BZOJ2402]陶陶的难题II(树链剖分+线段树维护凸包+分数规划)

    陶陶的难题II 时间限制:40s      空间限制:128MB 题目描述 输入格式 第一行包含一个正整数N,表示树中结点的个数. 第二行包含N个正实数,第i个数表示xi (1<=xi<= ...

  2. bzoj 2402: 陶陶的难题II 二分答案维护凸包

    2402: 陶陶的难题II Time Limit: 40 Sec  Memory Limit: 128 MBSec  Special JudgeSubmit: 68  Solved: 45[Submi ...

  3. BZOJ 3672[NOI2014]购票(树链剖分+线段树维护凸包+斜率优化) + BZOJ 2402 陶陶的难题II (树链剖分+线段树维护凸包+分数规划+斜率优化)

    前言 刚开始看着两道题感觉头皮发麻,后来看看题解,发现挺好理解,只是代码有点长. BZOJ 3672[NOI2014]购票 中文题面,题意略: BZOJ 3672[NOI2014]购票 设f(i)f( ...

  4. 【BZOJ2401】陶陶的难题I 欧拉函数+线性筛

    [BZOJ2401]陶陶的难题I 题意:求,n<=1000000,T<=100000 题解:直接做是n*sqrt(n)的,显然会TLE,不过这题a和b都是循环到n,那么就可以进行如下的神奇 ...

  5. bzoj 2401: 陶陶的难题I 数论

    2401: 陶陶的难题I Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 89  Solved: 24[Submit][Status] Descript ...

  6. P1676陶陶吃苹果 - vijos

    描述 curimit知道陶陶很喜欢吃苹果.于是curimit准备在陶陶生日的时候送给他一棵苹果树. curimit准备了一棵这样的苹果树作为生日礼物:这棵苹果树有n个节点,每个节点上有c[i]个苹果, ...

  7. 武汉科技大学ACM:1007: 陶陶摘苹果

    Problem Description 厘米高的板凳,当她不能直接用手摘到苹果的时候,就会踩到板凳上再试试. 个苹果到地面的高度,以及陶陶把手伸直的时候能够达到的最大高度,请帮陶陶算一下她能够摘到的苹 ...

  8. 洛谷-陶陶摘苹果(升级版)-BOSS战-入门综合练习1

    题目描述 Description 又是一年秋季时,陶陶家的苹果树结了n个果子.陶陶又跑去摘苹果,这次她有一个a公分的椅子.当他手够不着时,他会站到椅子上再试试. 这次与NOIp2005普及组第一题不同 ...

  9. NOIP2005-普及组复赛-第一题-陶陶摘苹果

    题目描述 Description 陶陶家的院子里有一棵苹果树,每到秋天树上就会结出10个苹果.苹果成熟的时候,陶陶就会跑去摘苹果.陶陶有个30厘米高的板凳,当她不能直接用手摘到苹果的时候,就会踩到板凳 ...

  10. noip普及组2005 陶陶摘苹果

    陶陶摘苹果 描述 陶陶家的院子里有一棵苹果树,每到秋天树上就会结出10个苹果.苹果成熟的时候,陶陶就会跑去摘苹果.陶陶有个30厘米高的板凳,当她不能直接用手摘到苹果的时候,就会踩到板凳上再试试. 现在 ...

随机推荐

  1. Linux 安装Zookeeper<集群版>(使用Mac远程访问)

    阅读本文需要先阅读安装Zookeeper<准备> 一 架构细节 zookeeper集群根据投票选举的机制 选出leader和follower zookeeper集群节点建议是奇数 这里我准 ...

  2. Siki_Unity_2-4_UGUI_Unity5.1 UI 案例学习

    Unity 2-4 UGUI Unity5.1 UI 案例学习 任务1-1:UGUI简介 什么是GUI: 游戏的开始菜单 RPG游戏的菜单栏.侧边栏和功能栏(比如背包系统.任务列表等) 设计用来控制移 ...

  3. HBASE理论篇

    1.Hbase是什么 HBase是一种构建在HDFS之上的分布式.面向列的存储系统.在需要实时读写.随机访问超大规模数据集时,可以使用HBase. 尽管已经有许多数据存储和访问的策略和实现方法,但事实 ...

  4. Qt creator 最常用的13个快捷键

    alt +enter // 自动创建类的定义 F1 // 查看帮助,文档 F2 // 快速到变量声明 Shift + F2 // 函数的声明和定义之间快速切换 F4 // 在 cpp 和 h 文件切换 ...

  5. SQL判断是否存在

    判断数据库是否存在 ifexists(select*frommaster..sysdatabaseswherename=N’库名’) print’exists’ else print’notexist ...

  6. 无法设置主体sa的凭据

    设置允许SQL Server身份登录 1.先用Window方式登陆进去,选择数据库实例,右键选择属性——安全性:把服务器身份验证选项从“Window身份验证模式”改为“SQLServer和Window ...

  7. java面向对象的冒泡排序,选择排序和插入排序的比较

    这三种排序有俩个过程: 1.比较俩个数据. 2.交换俩个数据或复制其中一项. 这三种排序的时间级别 冒泡排序:比较 (N-1)+(N-2)+...+2+1 = N*(N-1)/2=N2/2 交换  0 ...

  8. c#程序的阅读

    1 .程序是为表示两个连续的整数不能被整除. 2 ,3 程序黑框得不出结果,所以不知道具体的结果和运行时间. 4 采用更好的专用电脑进行计算.

  9. 初识 es6之 const

    const声明一个只读的常量.一旦声明,常量的值就不能改变. 例子: const a=12; a=2;//报错,const 声明的是常量,不能改 const声明的变量不得改变值,这意味着,const一 ...

  10. QMultiMap使用

    版权声明:若无来源注明,Techie亮博客文章均为原创. 转载请以链接形式标明本文标题和地址: 本文标题:QMultiMap使用     本文地址:http://techieliang.com/201 ...