一个奇奇怪怪的复杂度很垃圾的线段树合并解法


通过分析可以发现,要找$x$的$k$辈兄弟,只需要找到$x$的$k$辈祖先,然后查找以该祖先为根的子树中和$x$深度相同的节点个数$-1$即可。也就是说,询问只与深度有关,与具体是哪个节点无关。

具体怎么实现呢?想到dfs处理,显然在遍历过$x$后会回溯到$x$的$k$辈祖先,因此有一个想法,就是在回溯到$x$的$k$辈祖先时执行查询对应的查询就可以了,这样是一个离线的做法。目前的问题就是,如何能在$x$的$k$辈祖先时执行查询。

有一个做法就是提前处理出$x$的祖先,然后把询问插入到$x$的$k$辈祖先处,等到回溯到的时候进行查询;另一种就是一种比较奇奇怪怪的做法,利用大根堆维护询问,把祖先节点的深度作为关键字,当遍历到一个节点时,处理深度相同的询问。具体询问用线段树合并就可以处理了。

还有一个可以优化的地方,如果有祖先节点和查询节点的深度都相同的询问,答案相同,不需要重复查询。

#include<iostream>
#include<cstdio>
#include<vector>
#include<queue>
using namespace std;
const int N=1e5+100;
struct Seg
{
int lson,rson,sum;
#define lson(i) t[i].lson
#define rson(i) t[i].rson
#define sum(i) t[i].sum
}t[5000000];
struct Que
{
int fa,son,id;//祖先深度,查询节点深度,询问编号
};
int n,m,tot,cnt;
int head[N],ver[N],Next[N];
int r[N],ans[N];
bool v[N];
priority_queue<Que> q;//维护询问
vector<pair<int,int> > query[N];//存储询问
bool operator<(const Que &a,const Que &b)
{
return a.fa==b.fa?a.son<b.son:a.fa<b.fa;//注意是小于号
}//大根堆
void add(int x,int y)
{
ver[++tot]=y,Next[tot]=head[x],head[x]=tot;
}
int merge(int x,int y)
{
if(!x)
return y;
if(!y)
return x;
sum(x)+=sum(y);
lson(x)=merge(lson(x),lson(y));
rson(x)=merge(rson(x),rson(y));
return x;
}//线段树合并
int ask(int &p,int l,int r,int k)
{
if(!p)
p=++cnt;
if(l==r)
return sum(p);
int mid=(l+r)>>1;
if(k<=mid)
return ask(lson(p),l,mid,k);
else
return ask(rson(p),mid+1,r,k);
}//查询
void change(int &p,int l,int r,int k)
{
if(!p)
p=++cnt;
if(l==r)
{
sum(p)++;
return ;
}
int mid=(l+r)>>1;
if(k<=mid)
change(lson(p),l,mid,k);
else
change(rson(p),mid+1,r,k);
}//修改
void solve(int x,int d)
{
v[x]=1;//访问数组
for(int i=head[x];i;i=Next[i])
{
int y=ver[i];
solve(y,d+1);
r[x]=merge(r[x],r[y]);//子节点合并
}
while(q.size() && q.top().fa==d)//处理深度相同的询问
{
int y=q.top().son,id=q.top().id;q.pop();
ans[id]=ask(r[x],1,1e5,y)-1;//查询
while(q.size() && q.top().fa==d && q.top().son==y)
{
ans[q.top().id]=ans[id];
q.pop();
}//询问等价不用重复查询
}
change(r[x],1,1e5,d);//修改
for(int i=0;i<query[x].size();i++)
if(d-query[x][i].first>0)//保证询问合法
{
Que now;
now.fa=d-query[x][i].first;
now.son=d;
now.id=query[x][i].second;
q.push(now);
}//将询问插入大根堆
}
int main()
{
scanf("%d",&n);cnt=n;
for(int i=1,x;i<=n;i++)
{
scanf("%d",&x);r[i]=i;
if(x)
add(x,i);
}
scanf("%d",&m);
for(int i=1,x,y;i<=m;i++)
{
scanf("%d%d",&x,&y);
query[x].push_back(make_pair(y,i));//插入询问
}
for(int i=1;i<=n;i++)
if(!v[i])
solve(i,1);
for(int i=1;i<=m;i++)
printf("%d ",ans[i]);
return 0;
}

CF208E Blood Cousins 题解的更多相关文章

  1. CF208E Blood Cousins

    Blood Cousins 题目描述 小C喜欢研究族谱,这一天小C拿到了一整张族谱. 小C先要定义一下k-祖先. x的1-祖先指的是x的父亲 x的k-祖先指的是x的(k-1)-祖先的父亲 小C接下来要 ...

  2. CF208E Blood Cousins(DSU,倍增)

    倍增求出祖先,\(\text{DSU}\)统计 本来想用树剖求\(K\)祖,来条链复杂度就假了 #include <cstring> #include <cstdio> #in ...

  3. CF 208E - Blood Cousins dfs序+倍增

    208E - Blood Cousins 题目:给出一棵树,问与节点v的第k个祖先相同的节点数有多少个. 分析: 寻找节点v的第k个祖先,这不就是qtree2简化版吗,但是怎么统计该祖先拥有多少个深度 ...

  4. [cf contest246] E - Blood Cousins Return

    [cf contest246] E - Blood Cousins Return time limit per test 3 seconds memory limit per test 256 meg ...

  5. Codeforces 246E - Blood Cousins Return (树上启发式合并)

    246E - Blood Cousins Return 题意 给出一棵家谱树,定义从 u 点向上走 k 步到达的节点为 u 的 k-ancestor,每个节点有名字,名字不唯一.多次查询,给出 u k ...

  6. Codeforces 208E - Blood Cousins(树上启发式合并)

    208E - Blood Cousins 题意 给出一棵家谱树,定义从 u 点向上走 k 步到达的节点为 u 的 k-ancestor.多次查询,给出 u k,问有多少个与 u 具有相同 k-ance ...

  7. Codeforces 246E Blood Cousins Return(树上启发式合并)

    题目链接 Blood Cousins Return #include <bits/stdc++.h> using namespace std; #define rep(i, a, b) f ...

  8. 【CF208E】Blood Cousins

    题目大意:给定一个 N 个点的森林,M 个询问,每次询问对于点 u 来说,有多少个点和 u 有相同的 K 级祖先. 题解:线段树合并适合处理子树贡献的问题. 发现要回答这个询问在点 u 处计算很困难, ...

  9. CF 208E. Blood Cousins [dsu on tree 倍增]

    题意:给出一个森林,求和一个点有相同k级祖先的点有多少 倍增求父亲然后和上题一样还不用哈希了... #include <iostream> #include <cstdio> ...

随机推荐

  1. Python time altzone()方法

    描述 Python time altzone() 函数返回格林威治西部的夏令时地区的偏移秒数.高佣联盟 www.cgewang.com 如果该地区在格林威治东部会返回负值(如西欧,包括英国).对夏令时 ...

  2. Python 字典(Dictionary) items()方法

    描述 Python 字典(Dictionary) items() 函数以列表返回可遍历的(键, 值) 元组数组.高佣联盟 www.cgewang.com 语法 items()方法语法: dict.it ...

  3. PHP date_offset_get() 函数

    ------------恢复内容开始------------ 实例 返回奥斯陆(在欧洲挪威)冬天和夏天相对于 UTC 的以秒计的时区偏移量: <?php$winter=date_create(& ...

  4. 牛客练习赛64 红色的樱花 exgcd 贪心

    LINK:The red sakura 暴怒狂樱 血染京都. 这题质量不咋地 这题也没啥营养. 不过还是存在值得学习的地方的. 一个trick n行 m列 第一行与第n行相连 第1列和第m列相连的时候 ...

  5. P3239 [HNOI2015]亚瑟王 期望 dp

    LINK:亚瑟王 Saber!Excalibur! 比较难的期望dp. 可以发现如果暴力枚举所有的局面复杂度很高 . 转换的思路则是 期望的线性性. 求出每张牌的期望累加即可. 考虑每张牌的期望=这张 ...

  6. Fragment为什么须要无参构造方法

    日前在项目代码里遇到偷懒使用重写Fragment带参构造方法来传参的做法,顿生好奇,继承android.support.v4.app.Fragment而又不写无参构造方法不是会出现lint错误编译不通 ...

  7. Armv8-A Memory management

    本文介绍Armv8-A的内存管理.内存管理指的是在系统中,内存访问是如何实现的. 使用内存管理机制,可以让每个应用之间的内存地址分离,即sandbox application,也可以让多个在物理内存上 ...

  8. 【NOI2018】归程 题解(kruskal重构树+最短路)

    题目链接 题目大意:给定一张$n$个点$m$条边的无向图.每条边有长度和海拔.有$Q$次询问,每次给定起点$v$和当天水位线$p$,每次终点都是$1$.人可以选择坐车或走路,车只能在海拔大于水位线的路 ...

  9. 【FZYZOJ】「Paladin」瀑布 题解(期望+递推)

    题目描述 CX在Minecraft里建造了一个刷怪塔来杀僵尸.刷怪塔的是一个极高极高的空中浮塔,边缘是瀑布.如果僵尸被冲入瀑布中,就会掉下浮塔摔死.浮塔每天只能工作 $t$秒,刷怪笼只能生成 $N$  ...

  10. Idea 提交配置说明

    Idea 提交配置说明# Auto-update after commit :自动升级后提交 keep files locked :把文件锁上,我想这应该就只能你修改其他开发人不能修改不了的功能 在你 ...