题目描述:

bz

luogu

题解:

主席树维护大力树剖。

一条路径上不允许过的点的个数是当前袭击数-$y$时袭击数,

所以允许经过的点的个数是总数-当前袭击数+$y$时袭击数。

用主席树去维护每个时刻树链袭击数。

这样的话修改的时间复杂度是$O(nlogn)$的。

现在的问题是查询。

注意端点不算路径上的点,那么我们可以让起点和终点距离近一点。

然后有两种情况,起点先向上走/先向下走。

这两个都能用同样的暴跳二分解决。

上代码:

#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int N = 100050;
const int M = 80*N;
template<typename T>
inline void read(T&x)
{
T f = 1,c = 0;char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
while(ch>='0'&&ch<='9'){c=c*10+ch-'0';ch=getchar();}
x = f*c;
}
int n,m,hed[N],cnt,rot;
struct EG{int to,nxt;}e[N];
void ae(int f,int t)
{
e[++cnt].to = t;
e[cnt].nxt = hed[f];
hed[f] = cnt;
}
int dep[N],siz[N],top[N],fa[N],son[N],tin[N],tim,pla[N];
void dfs0(int u,int f)
{
fa[u] = f,siz[u] = 1,dep[u] = dep[f]+1;
for(int j=hed[u];j;j=e[j].nxt)
{
int to = e[j].to;
dfs0(to,u);siz[u]+=siz[to];
if(siz[to]>siz[son[u]])son[u]=to;
}
}
void dfs1(int u,int Top)
{
top[u] = Top,tin[u] = ++tim,pla[tim] = u;
if(son[u])dfs1(son[u],Top);
for(int j=hed[u];j;j=e[j].nxt)
{
int to = e[j].to;
if(to!=son[u])dfs1(to,to);
}
}
int get_lca(int x,int y)
{
while(top[x]!=top[y]){if(dep[top[x]]<dep[top[y]])swap(x,y);x=fa[top[x]];}
return dep[x]<dep[y]?x:y;
}
int get_son(int x,int y)
{
while(top[x]!=top[y]){if(fa[top[x]]==y)return top[x];x=fa[top[x]];}
return pla[tin[y]+1];
}
int rt[N];
struct pertree
{
int tot,ls[M],rs[M],siz[M];
void insert(int l,int r,int&u,int las,int qx)
{
u = ++tot;
ls[u] = ls[las],rs[u] = rs[las],siz[u] = siz[las]+1;
if(l==r)return ;
int mid = (l+r)>>1;
if(qx<=mid)insert(l,mid,ls[u],ls[las],qx);
else insert(mid+1,r,rs[u],rs[las],qx);
}
int query(int l,int r,int L,int R,int ql,int qr)
{
if(!L&&!R)return 0;
if(l==ql&&r==qr)return siz[R]-siz[L];
int mid = (l+r)>>1;
if(qr<=mid)return query(l,mid,ls[L],ls[R],ql,qr);
else if(ql>mid)return query(mid+1,r,rs[L],rs[R],ql,qr);
else return query(l,mid,ls[L],ls[R],ql,mid)+query(mid+1,r,rs[L],rs[R],mid+1,qr);
}
}tr;
int query(int x,int y,int las,int now)
{
int ret = 0;
while(top[x]!=top[y])
{
ret+=dep[x]-dep[top[x]]+1-tr.query(1,n,rt[las],rt[now],tin[top[x]],tin[x]);
x = fa[top[x]];
}
return ret+dep[x]-dep[y]+1-tr.query(1,n,rt[las],rt[now],tin[y],tin[x]);
}
int dv(int l,int r,int las,int now,int k)
{
int ret = l-1;int tmp = r;
while(l<=r)
{
int mid = (l+r)>>1;
if(tmp-mid+1-tr.query(1,n,rt[las],rt[now],mid,tmp)>=k)ret=mid,l=mid+1;
else r=mid-1;
}
return pla[ret];
}
int jmp(int x,int y,int las,int now,int k)
{
while(top[x]!=top[y])
{
int tmp = dep[x]-dep[top[x]]+1-tr.query(1,n,rt[las],rt[now],tin[top[x]],tin[x]);
if(k>tmp)k-=tmp,x=fa[top[x]];
else return dv(tin[top[x]],tin[x],las,now,k);
}
return dv(tin[y],tin[x],las,now,k);
}
int main()
{
// freopen("tt.in","r",stdin);
read(n);
for(int f,i=1;i<=n;i++)
{
read(f);
if(!f)rot=i;
else ae(f,i);
}
dfs0(rot,0),dfs1(rot,rot);
read(m);
for(int op,a,b,c,k,y,i=1;i<=m;i++)
{
read(op);
if(op==1)
{
read(c);
tr.insert(1,n,rt[i],rt[i-1],tin[c]);
}else
{
rt[i]=rt[i-1];
read(a),read(b),read(k),read(y);
if(fa[a]==b||fa[b]==a){puts("-1");continue;}
int Lca = get_lca(a,b);
if(a!=Lca&&b!=Lca)a=fa[a],b=fa[b];
else if(a==Lca)b=fa[b],a=get_son(b,a);
else a=fa[a],b=get_son(a,b);
Lca = get_lca(a,b);
if(a==Lca)
{
int sum = query(b,a,y,i);
if(sum<k){puts("-1");continue;}
printf("%d\n",jmp(b,a,y,i,sum-k+1));
}else
{
int alc = get_son(a,Lca);
int sum = query(a,alc,y,i);
if(sum>=k){printf("%d\n",jmp(a,alc,y,i,k));continue;}
k-=sum;
sum = query(b,Lca,y,i);
if(sum<k){puts("-1");continue;}
printf("%d\n",jmp(b,Lca,y,i,sum-k+1));
}
}
}
return 0;
}

CF226E Noble Knight's Path/bzoj4704 旅行的更多相关文章

  1. [CF226E]Noble Knight's Path

    [CF226E]Noble Knight's Path 题目大意: 一棵\(n(n\le10^5)\)个结点的树,初始时所有结点都是白色.\(m(m\le10^5)\)次操作,操作包含以下两种: 将点 ...

  2. [Codeforces 226E]Noble Knight's Path

    题目大意:有一棵n个节点的树,m年.初始每个节点都有.每天有如下操作:1. 给定c,让c没有(c只可能没有一次).2. 给定s,t,k,y,求从第y+1年到现在(即忽略y+1年之前的操作1),s到t的 ...

  3. Lintcode: Knight Shortest Path

    Given a knight in a chessboard (a binary matrix with 0 as empty and 1 as barrier) with a source posi ...

  4. The Sorrows of Young Werther

    The Sorrows of Young Werther J.W. von Goethe Thomas Carlyle and R.D. Boylan Edited by Nathen Haskell ...

  5. 九章lintcode作业题

    1 - 从strStr谈面试技巧与代码风格 必做题: 13.字符串查找 要求:如题 思路:(自写AC)双重循环,内循环读完则成功 还可以用Rabin,KMP算法等 public int strStr( ...

  6. 宽度优先搜索(BFS)— 20180909 - 20180917

    BFS几类题: 1.图的遍历:a.层级遍历 b.由点及面 c.拓扑排序 2.简单图最短路径: 简单图:1.无向图 2.边权重一致 图的时间复杂度: N个点,M条边,M最大是N^2,时间复杂度O(N+M ...

  7. BFS算法的优化 双向宽度优先搜索

    双向宽度优先搜索 (Bidirectional BFS) 算法适用于如下的场景: 无向图 所有边的长度都为 1 或者长度都一样 同时给出了起点和终点 以上 3 个条件都满足的时候,可以使用双向宽度优先 ...

  8. A Child's History of England.22

    CHAPTER 8 ENGLAND UNDER WILLIAM THE FIRST, THE NORMAN CONQUEROR Upon the ground where the brave Haro ...

  9. 【BZOJ4704】旅行 树链剖分+可持久化线段树

    [BZOJ4704]旅行 Description 在Berland,有n个城堡.每个城堡恰好属于一个领主.不同的城堡属于不同的领主.在所有领主中有一个是国王,其他的每个领主都直接隶属于另一位领主,并且 ...

随机推荐

  1. 在这个插件帮助下,终于用上免费的Https协议外链的图床了

    前天,强哥发了一篇推文,讲述了应该如何免费且快速的生成自己的博客网站: 期间也有提到一点就是我们在写博客的时候,因为使用的是Markdown格式的文件,而如果想要​Markdown格式的文件在图片上传 ...

  2. CreateEvent进程同步

    CreateEvent进程间同步   CreateEvent可以创建或是打开一个命名或是未命名的event对象. HANDLE CreateEvent(   LPSECURITY_ATTRIBUTES ...

  3. python 2048游戏控制器

    2048游戏控制器 1 evaluate 要用程序来处理就得对现实的问题进行量化,用数字来表示.在2048游戏中,我们的输入是一个棋局,让我们输出一个移动方向,这样我们需要对棋局进行量化,即我们要评估 ...

  4. 后台运行程序-服务器、python

    0前言 最近遇到一个需求,我有一个很小的python程序,需要一直在服务器器上跑,但是我不想一直开着浏览器或者Xshell 7,所以记录一下怎么解决的. 用到的指令是nohup 具体代码就两行 sou ...

  5. ARM7、ARM9、ARM11、ARM-Cortex系列的关系

    参考资料: https://zhuanlan.zhihu.com/p/92315825 https://zhuanlan.zhihu.com/p/82337495 ARM是Advanced RISC ...

  6. 『无为则无心』Python面向对象 — 58、类方法和静态方法

    目录 1.实例方法 2.类方法 (1)类方法特点 (2)类方法使用场景 3.静态方法 (1)静态方法特点 (2)静态方法使用场景 1.实例方法 实例方法既可以调用静态方法也可以调用类方法. # 定义一 ...

  7. 盘点Go中的开发神器

    本文已收录 https://github.com/lkxiaolou/lkxiaolou 欢迎star. 在Java中,我们用Junit做单元测试,用JMH做性能基准测试(benchmark),用as ...

  8. java策略模式拙见

    面向对象的两个基本准则: 单一职责:一个类只有一个发生变化的原因 开闭原则:对拓展开放,对修改关闭 <Java开发手册>中,有这样的规则:超过3层的 if-else 的逻辑判断代码可以使用 ...

  9. oj教程--坑

    1.OJ判断是只看输出结果的. 2.纯字符串用puts()输出. 3.有很多数学题是有规律的,直接推公式或用递归.循环. 4.擅用三目运算符 5.将乘法转换成加法减少时间 6.空间换时间 7.数组越界

  10. ts转js运行报错:“tsc : 无法加载文件

    一.在typescript.ts转换成.js运行时报错解决办法: 1.第一步:鼠标在vscode软件上右击打开属性–>兼容性–>以管理员的身份运行此程序,如下图: 2.第二步:打开vsco ...