[UOJ198]时空旅行
看懂题目就知道$y,z$是没用的,这题相当于是给一堆$(x_i,c_i)$和询问$x_q$,要求$(x_q-x_i)^2+c_i$的最大值
先把这个式子拆开:$-2x_ix_q+x_i^2+c_i+x_q^2$,那么询问就是求一堆直线$y=-2x_ix+x_i^2+c_i$在$x=x_q$处的最小值,维护坐标轴底端的上凸壳即可
再看题目中关于“时空”的限制,其实就是给一棵树,某一些点会有标记表示以这个点为根的子树内有/没有一条直线,我们用dfs序在线段树上覆盖对应区间并预处理出每个线段树节点的凸壳即可
把询问按$x$排序,那么我们查询时就可以在线段树上利用单调性做到$O(m\log_2n+n)$
总时间复杂度$O(n\log_2^2n+m\log_2n)$
代码准确度++
#include<stdio.h>
#include<algorithm>
#include<vector>
using namespace std;
typedef long long ll;
typedef double du;
const du eps=1e-8;
const ll inf=9223372036854775807ll;
int h[500010],nex[500010],to[500010],in[500010],ou[500010],M;
void add(int a,int b){
M++;
to[M]=b;
nex[M]=h[a];
h[a]=M;
}
void dfs(int x){
M++;
in[x]=M;
for(int i=h[x];i;i=nex[i])dfs(to[i]);
ou[x]=M;
}
ll c[500010],x[500010];
int at[500010];
vector<int>pl[2000010],del[500010],tmp;
vector<int>::iterator it;
bool cmp1(int x,int y){return in[x]<in[y];}
bool cmp2(int i,int j){return(x[i]==x[j])?x[i]*x[i]+c[i]>x[j]*x[j]+c[j]:(x[i]<x[j]);}
void modify(int L,int R,int v,int l,int r,int x){
if(L<=l&&r<=R)return pl[x].push_back(v);
int mid=(l+r)>>1;
if(L<=mid)modify(L,R,v,l,mid,x<<1);
if(mid<R)modify(L,R,v,mid+1,r,x<<1|1);
}
du ins(int i,int j){return(x[i]*x[i]+c[i]-x[j]*x[j]-c[j])/((x[i]-x[j])*2.);}
bool leq(du x,du y){return x-y<eps;}
void dfs(int l,int r,int u){
sort(pl[u].begin(),pl[u].end(),cmp2);
tmp.clear();
for(it=pl[u].begin();it!=pl[u].end();it++){
while(!tmp.empty()){
if(x[*tmp.rbegin()]==x[*it]){
tmp.pop_back();
continue;
}
if(tmp.size()==1)break;
if(leq(ins(*tmp.rbegin(),*it),ins(tmp[tmp.size()-2],*it)))
tmp.pop_back();
else
break;
}
tmp.push_back(*it);
}
pl[u]=tmp;
if(l==r)return;
int mid=(l+r)>>1;
dfs(l,mid,u<<1);
dfs(mid+1,r,u<<1|1);
}
struct ask{
int s,id;
ll x;
}q[500010];
ll ans[500010];
int head[2000010];
bool cmp3(ask a,ask b){return a.x<b.x;}
ll calc(int i,ll x0){return-2ll*x[i]*x0+x[i]*x[i]+c[i];}
ll query(int u,ll x0,int l,int r,int x){
ll ans=inf;
if(!pl[x].empty()){
while(head[x]<(int)pl[x].size()-1&&leq(ins(pl[x][head[x]],pl[x][head[x]+1]),x0))head[x]++;
ans=min(ans,calc(pl[x][head[x]],x0));
}
if(l==r)return ans;
int mid=(l+r)>>1;
if(u<=mid)
ans=min(ans,query(u,x0,l,mid,x<<1));
else
ans=min(ans,query(u,x0,mid+1,r,x<<1|1));
return ans;
}
int main(){
int n,m,i,op,fr,id,y,z;
scanf("%d%d%lld",&n,&m,c+1);
at[1]=1;
for(i=2;i<=n;i++){
scanf("%d%d%d",&op,&fr,&id);
fr++;
id++;
add(fr,i);
if(op==0){
scanf("%lld%d%d%lld",x+id,&y,&z,c+id);
at[id]=i;
}else
del[id].push_back(i);
}
M=0;
dfs(1);
for(i=1;i<=n;i++){
if(at[i]){
sort(del[i].begin(),del[i].end(),cmp1);
y=in[at[i]];
for(it=del[i].begin();it!=del[i].end();it++){
if(y<in[*it])modify(y,in[*it]-1,i,1,n,1);
y=ou[*it]+1;
}
if(y<=ou[at[i]])modify(y,ou[at[i]],i,1,n,1);
}
}
dfs(1,n,1);
for(i=1;i<=m;i++){
scanf("%d%lld",&q[i].s,&q[i].x);
q[i].s++;
q[i].id=i;
}
sort(q+1,q+m+1,cmp3);
for(i=1;i<=m;i++)ans[q[i].id]=query(in[q[i].s],q[i].x,1,n,1)+q[i].x*q[i].x;
for(i=1;i<=m;i++)printf("%lld\n",ans[i]);
}
[UOJ198]时空旅行的更多相关文章
- @loj - 2987@ 「CTSC2016」时空旅行
目录 @description@ @solution@ @accepted code@ @details@ @description@ 2045 年,人类的技术突飞猛进,已经找到了进行时空旅行的方法. ...
- [UOJ198][CTSC2016]时空旅行
uoj description 你要维护若干个集合,每个集合都是有一个编号比他小的集合扩展而来,扩展内容为加入一个新的元素\((x,c)\)或者删除一个已有元素.集合的扩展关系之间构成一个树形结构. ...
- uoj198【CTSC2016】时空旅行
传送门:http://uoj.ac/problem/198 [题解] 首先y.z是没有用的.. 然后式子就是w = (x0-xi)^2+ci的最小值,化出来可以变成一个直线的形式. 然后我们可以用线段 ...
- 【UOJ #198】【CTSC 2016】时空旅行
http://uoj.ac/problem/198 (先补一下以前的题解) 这道题5分暴力好写好调,链上部分分可以用可持久化线段树,每次旅行\(x\)值相同的可以用标记永久化线段树.我还听到某些神犇说 ...
- 【CTSC2016】时空旅行
链接 http://uoj.ac/problem/198 题解 首先要发现答案要我们求这个式子: \[ ans=min\bigl((x_i-x)^2+c_i\bigr) \] 显而易见的是这种时空嫁接 ...
- Luogu P5416 [CTSC2016]时空旅行
第一次写线段树分治的题目,没想到是道这么毒的题233 首先发现题目里的\((x,y,z,c)\)就是在放屁,只有\((x,c)\)是有用的 因此我们可以把题意转化为,在某一个时间节点上,求出所有元素的 ...
- [CTSC2016]时空旅行
description 题面 solution 线段树分治+斜率优化毒瘤题 题目可以简化为: 你要维护一个包含元素\((x,c)\)的集合 修改操作为从以前的一个版本更新,修改内容为添加或删除一个元素 ...
- [CTSC2016]时空旅行(线段树+凸包)
应该是比较套路的,但是要A掉仍然不容易. 下面理一下思路,思路清楚了也就不难写出来了. 0.显然y,z坐标是搞笑的,忽略即可. 1.如果x不变,那么直接set即可解决. 2.考虑一个空间和询问x0,通 ...
- CTSC2016时空旅行
当时看这道题AC的人数比较多,就开了这道题. 很容易发现是这是一个有关凸包的题. 然后不知道怎么维护凸包,一直在想cdq,感觉复杂度不行,于是被这玩意难住了…… 幸好有亲学长yyh造福人类的题解:ht ...
随机推荐
- JS遮罩层弹框效果
对于前端开发者来说,js是不可缺少的语言.现在我开始把我日常积累的一些js效果或者通过搜索自己总结的一些效果分享给大家,希望能够帮助大家一起进步,也希望大家能够多多支持! 1.今天我先分享一个遮罩层弹 ...
- jsp中的路径问题
在学jsp的时候我每次都遇到路径的问题,每次都不能够说100%的把这搞定,这让我很烦恼,今天下午花了点时间来把路径问题整理了下. 一:首先我们在加载项目(我的项目名称是FinalExam)是的路径是h ...
- nginx proxy_buffer_size 解决后端服务传输数据过多,其实是header过大的问题
nginx proxy_buffer_size 解决后端服务传输数据过多,其实是header过大的问题 这三个参数已设置就搞定了额 proxy_buffer_size 64k; proxy_buffe ...
- linux编译动态库 fPIC作用
在生成动态库时,常常习惯性的加上fPIC选项,fPIC有什么作用和意义,加不加有什么区别,这里做下小结: fPIC的全称是 Position Independent Code, 用于生成位置无关代码. ...
- saltstack入门至放弃之salt安装部署
学习了一段时间的saltstack,是时候记录下了.友提:学习环境是两台centos_7.2_x64机器 系统初始化: 两台机器执行以下脚本即可(友提:两台服务器的主机名配置在/etc/hosts中, ...
- OpenStack环境初始化
环境概述 系统:CentOS_7.2_x64_mininal 因配置有限,本次试验使用三台虚拟机,一台控制节点,一台计算节点,一台网络节点,控制机点配置4G内存,2CPU,其他节点都2G内存,一个C ...
- bzoj1833: [ZJOI2010]count 数字计数 && codevs1359 数字计数
bzoj1833 codevs1359 这道题也是道数位dp 因为0有前导0这一说卡了很久 最后发现用所有位数减1~9的位数就okay.....orzczl大爷 其他就跟51nod那道统计1出现次数一 ...
- [bzoj4515][Sdoi2016]游戏-树链剖分+李超线段树
Brief Description Alice 和 Bob 在玩一个游戏. 游戏在一棵有 n 个点的树上进行.最初,每个点上都只有一个数字,那个数字是 123456789123456789. 有时,A ...
- 入园的第一篇--where、where
这篇是入园的第一篇随便,后面我会将自己几年前写的博文都转到这里.哎,其实说到博文的事情,我就很郁闷,甚至有些恼火,后面我会详细说说这中间的过程,也许能帮助某些人避免遇到类似的事情.突然想起<西游 ...
- Linux虚拟地址空间布局以及进程栈和线程栈总结【转】
转自:http://www.cnblogs.com/xzzzh/p/6596982.html 原文链接:http://blog.csdn.net/freeelinux/article/details/ ...