[CTSC2016]时空旅行

description

题面

solution

线段树分治+斜率优化毒瘤题

题目可以简化为:

你要维护一个包含元素\((x,c)\)的集合

修改操作为从以前的一个版本更新,修改内容为添加或删除一个元素

查询操作给出\(x_0\),查询某个版本中的\(max\{(x-x_0)^2+c\}\)

可以知道版本之间的时间关系形成一颗树

如果在一个版本删除了某个元素,那么在这个版本的子树中都不会再有这个版本

由于子树的\(dfn\)是连续的,因此操作可以简化为在序列上进行,总共有\(O(m)\)个区间

最简单的方法是对于线段树上的每个节点开\(vector\)<\(query\)>和\(vector\)<\(modify\)>,

注意区间查询和单点修改的插入方式有所不同

斜率优化的式子就是\(max_i\{x_i^2+c_i-2x_0x_i\}+x_0^2\)

然而这题比较丧,它会给出两个横坐标相等的点

因此一定要判断有向斜率

我们先将修改和询问按照\(x\)和\(k\)外部排序,这样可以保证线段树分治时\(x\)和\(k\)都单调

于是再每个线段树节点里可以单调队列直接做

最后这题卡空间

于是我们只插入修改,对于询问直接递归求解即可

总复杂度为\(O((n+m)logn)\)

code

#include<bits/stdc++.h>
#include<algorithm>
#include<iostream>
#include<cstdlib>
#include<iomanip>
#include<cstring>
#include<complex>
#include<vector>
#include<cstdio>
#include<string>
#include<bitset>
#include<ctime>
#include<cmath>
#include<queue>
#include<stack>
#include<map>
#include<set>
#define FILE "a"
#define mp make_pair
#define pb push_back
#define RG register
#define il inline
using namespace std;
typedef unsigned long long ull;
typedef vector<int>VI;
typedef long long ll;
typedef double dd;
const dd eps=1e-10;
const int mod=1e9+7;
const int N=5e5+10;
const int T=5e6+10;
const dd pi=acos(-1);
const int inf=2147483647;
const ll INF=1e18+1;
il ll read(){
    RG ll data=0,w=1;RG char ch=getchar();
    while(ch!=‘-‘&&(ch<‘0‘||ch>‘9‘))ch=getchar();
    if(ch==‘-‘)w=-1,ch=getchar();
    while(ch<=‘9‘&&ch>=‘0‘)data=data*10+ch-48,ch=getchar();
    return data*w;
}
il void file(){
    srand(time(NULL)+rand());
    freopen(FILE".in","r",stdin);
    freopen(FILE".out","w",stdout);
}

int n,m,mt,b[N],s[N];ll x[N],y[N],ans[N];
struct modify{int l,r;ll x,y;}M[N];
bool cmpm(modify a,modify b){return a.x<b.x;}
struct query{int id,s;ll x;}Q[N];
bool cmpq(query a,query b){return a.x<b.x;}
int head[N],nxt[N<<1],to[N<<1],cnt;
il void add(int u,int v){
    to[++cnt]=v;
    nxt[cnt]=head[u];
    head[u]=cnt;
}

VI ML[N],MR[N];int w[N],cntw,sz[N];
void dfs(int u,int fa){
    w[u]=++cntw;sz[u]=1;
    if(!b[u])ML[s[u]].pb(w[u]);
    else MR[s[u]].pb(w[u]-1);
    for(RG int i=head[u];i;i=nxt[i]){
        RG int v=to[i];if(v==fa)continue;
        dfs(v,u);sz[u]+=sz[v];
    }
    if(!b[u])MR[s[u]].pb(w[u]+sz[u]-1);
    else ML[s[u]].pb(w[u]+sz[u]);
}

VI f[N<<2];int top[N<<2];
#define mid ((l+r)>>1)
#define ls (i<<1)
#define rs (i<<1|1)
il double getk(ll ax,ll ay,ll bx,ll by){
    if(ax==bx)return ay>by?-inf:inf;
    return (ay-by)*1.0/(ax-bx);
}

void insertmodify(int i,int l,int r,int xl,int yr,int id){
    if(xl<=l&&r<=yr){
        RG int R=f[i].size()-1;RG ll qx=x[id],qy=y[id];
        while(R>0&&getk(x[f[i][R-1]],y[f[i][R-1]],x[f[i][R]],y[f[i][R]])>getk(x[f[i][R]],y[f[i][R]],qx,qy))R--,f[i].pop_back();
        R++;f[i].pb(id);return;
    }
    if(xl<=mid)insertmodify(ls,l,mid,xl,yr,id);
    if(mid<yr)insertmodify(rs,mid+1,r,xl,yr,id);
}

ll ask(int i,int l,int r,int p,query q){
    RG int &L=top[i],R=f[i].size()-1;RG ll ans=INF;
    while(L<R&&getk(x[f[i][L]],y[f[i][L]],x[f[i][L+1]],y[f[i][L+1]])<q.x)
        L++;
    if(L<=R)ans=y[f[i][L]]-q.x*x[f[i][L]]+q.x*q.x;
    if(l==r)return ans;
    if(p<=mid)ans=min(ans,ask(ls,l,mid,p,q));
    else ans=min(ans,ask(rs,mid+1,r,p,q));
    return ans;
}

int main()
{
    n=read();m=read();s[1]=1;y[1]=read();
    for(RG int i=2,o,f,id;i<=n;i++){
        o=read();f=read()+1;id=read()+1;add(f,i);add(i,f);b[i]=o;s[i]=id;
        if(!o){x[id]=read();read();read();y[id]=read();}
    }
    dfs(1,0);

    for(RG int i=1;i<=n;i++)
        for(RG int j=0,sz=ML[i].size();j<sz;j++)
            M[++mt]=(modify){ML[i][j],MR[i][j],2ll*x[i],1ll*x[i]*x[i]+y[i]};
    sort(M+1,M+mt+1,cmpm);
    for(RG int i=1;i<=mt;i++)x[i]=M[i].x,y[i]=M[i].y;
    for(RG int i=1;i<=m;i++)
        Q[i].id=i,Q[i].s=read()+1,Q[i].x=read(),ans[i]=INF;
    sort(Q+1,Q+m+1,cmpq);

    for(RG int i=1;i<=mt;i++)insertmodify(1,1,n,M[i].l,M[i].r,i);
    for(RG int i=1;i<=m;i++)ans[Q[i].id]=ask(1,1,n,w[Q[i].s],Q[i]);
    for(RG int i=1;i<=m;i++)printf("%lld\n",ans[i]);
    return 0;
}

原文地址:https://www.cnblogs.com/cjfdf/p/9383630.html

时间: 2024-07-30 15:38:50

[CTSC2016]时空旅行的相关文章

[CTSC2016]时空旅行(线段树+凸包)

应该是比较套路的,但是要A掉仍然不容易. 下面理一下思路,思路清楚了也就不难写出来了. 0.显然y,z坐标是搞笑的,忽略即可. 1.如果x不变,那么直接set即可解决. 2.考虑一个空间和询问x0,通过化式子发现实际上就是:把每个星球看成一个一次函数,其实是在询问这个空间内的所有一次函数在x0处的最小值. 3.这个显然是一个凸包,所以我们需要对每个空间维护一个凸包,由空间整体呈树状,可以想到用DFS序+线段树维护区间. 4.预处理出每个星球的存在范围,在线段树上永久化标记.查询时依次递归求最小值

luogu P5416 [CTSC2016]时空旅行

luogu uoj 注意到用这个集合产生方式可以构建出一个树型结构,并且每个加入/删除元素都是对应的一个子树的范围,对应到dfs序上就是每次对一个区间内的集合加入/删除元素,所以可以线段树分治,把每种元素的出现区间整出来 把答案柿子\((x-x_0)^2+c\)拆开,得到\(x^2-2x*x_0+{x_0}^2+c\),然后每个元素就相当于斜率为\(-2x\),截距为\(x^2+c\)的直线,所以线段树每个节点维护凸壳,要用的时候直接查对应横坐标的值.如果直接做是两个\(log\)的,但是因为要

Luogu P5416 [CTSC2016]时空旅行(线段树分治)

题目 简化题意:你需要维护\(n\)个集合,集合元素为二元组\((x,v)\).集合\(i\)的产生方式是以某个原有集合\(p_i\)为样本,扩展或删除一个元素后得到新集合.有\(q\)次询问,每次给出\(y\)并指定一个集合\(i\),要求从集合\(i\)中找出一个元素,最小化\((x?y)^2+v\). 先拆式子\((x-y)^2+v=x^2-2xy+y^2+v\),令其等于\(k\)即\(x^2+y^2-2xy+v=k\). 移项得\(2yx+k=y^2+x^2+v\),可以看作是\((x

@loj - [email&#160;protected] 「CTSC2016」时空旅行

目录 @[email protected] @[email protected] @accepted [email protected] @[email protected] @[email protected] 2045 年,人类的技术突飞猛进,已经找到了进行时空旅行的方法.小 R 得到了一台时空旅行仪,他想用它调查不同时空中人类的发展状况. 根据平行时空理论,宇宙中存在着很多独立的时空,每个时空在下一个时间点还会分化出若干个不同的时空.宇宙是一个三维空间,人类使用空间直角坐标系来描述空间中的

时空旅行的可能性(无聊研究社)

经济不好的时候,人一般都没有什么爱好,所以这段时间只能以看相对论啊,宇宙起源之类的东西拿来消遣,说实话这类消遣是最廉价的,成本不会高于宅男们欣赏苍老师们的表演的成本.仔细的阅读了一些这些大师们的著作后,我发现,时空旅行理论上可以,但实际上很难,因为: 1)如果靠速度去旅行,宇宙天体之间的距离动不动就是多少光年,也就是说光都要跑好多年,要实现早上出发晚上到啥恒星去吃晚饭的梦想,非常的困难,一是很难达到光的速度,就是达到了,也动不动就要好几年:所以要达到星际朝发夕至的梦想,靠速度是不行的.2)靠速度

uoj198【CTSC2016】时空旅行

传送门:http://uoj.ac/problem/198 [题解] 首先y.z是没有用的.. 然后式子就是w = (x0-xi)^2+ci的最小值,化出来可以变成一个直线的形式. 然后我们可以用线段树维护dfs序上的每个点. 每个点维护经过这个点的所有直线(标记永久化),也就是维护上凸壳. 然后我们把询问按照x排序,每次决策点只会后移.所以复杂度就有保证啦! 真**难写 还有一个十分有趣的事实啊 我用一个号交完ac在另一个号再交就RE了啊... 不管了反正过了 # include <vecto

[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序在线段树上覆盖对应区间并预处理出每个线段树节点的凸

YCB 的暑期计划

前言 YCB现在很弱(TAT) 暑假有一个月,赶快狂补一下. 大概的计划如下: 首先前期会以数据结构为主,毕竟代码能力太弱,涉及内容:线段树分治.二进制分组.KD-Tree. 等数据结构做到没有智商的时候加入一波数论,内容为 杜教筛.min_25筛. 然后中途小清新一下,做一些 组合博弈与构造题. 接着继续练代码能力,顺便学一些神奇的暴力:启发式合并.dsu on tree . 然后图论也忘的差不多了,就回过头去学点新东西,大概会有spfa判负环.0/1分数规划.差分约束. 估计这个时候也没有什

线段树分治总结

目录 类型一 例题1:八纵八横 代码: 例题2:时空旅行 首先,要求可以离线. 线段树分治有两种. 类型一 操作基于区间,单点询问. 有时,进行的一种操作可以快速完成,但是,要实现这种操作的逆操作较难. 因为,通常情况下,需要实现的逆操作都是很久以前执行的. 但是,如果只撤销上次操作,就会简单得多. 比如,维护一些连通性,或直径,线性基等问题. 这类问题加边很好做,但删边很难实现. 我们可以扫一遍操作,得到每个操作的有效区间. 然后,将每个添加操作的有效区间按在线段树上,然后遍历这颗线段树同时处