Blood Cousins

题目描述

小C喜欢研究族谱,这一天小C拿到了一整张族谱。

小C先要定义一下k-祖先。

  • x的1-祖先指的是x的父亲
  • x的k-祖先指的是x的(k-1)-祖先的父亲

小C接下来要定义k-兄弟

  • xx的k-兄弟指的是与x的k-祖先相同的人
  • 如果不存在k-祖先那么x没有k-兄弟

小C想问问你,xx到底有多少k-兄弟?小C打算问QQ次这样的问题。

输入格式

第一行11个数nn,表示族谱中的人数,从11开始编号。

接下来一行nn个数,第ii个数r_iri​指的是ii的父亲,如果为0,则可能是因为族谱数据不全,它在族谱中没有父亲。

接下来11个数QQ,表示小C问题的个数。

接下来QQ行,每行两个数x, kx,k,表示一个问题。

输出格式

输出一行QQ个数,表示问题的答案。


处理询问的话,可以把询问按dfs序排序,然后倒着处理

对于每个询问$(x,k)$,可以转化成:在以x的k-祖先为根的子树上,深度与$x$相同的点的个数

#include <cstdio>
#include <vector>
#include <algorithm>
using namespace std;
#define N 100005
void read(int &x) {
char c = getchar();
x = ;
while (c < '' || c > '') c = getchar();
while ('' <= c && c <= '') x = x * + c - , c = getchar();
}
int n, K, Q, siz[N], d[N], c[N], big[N], R;
int _S[N], tp, cok, dfn[N], fa[][N], ans[N];
struct Data {
int x, k, id;
} a[N];
bool cmp(Data A, Data B) { return dfn[A.x] < dfn[B.x]; }
bool vis[N];
vector<int> g[N]; void dfs1(int x, int Fa) {
siz[x] = ;
d[x] = d[Fa] + ;
fa[][x] = Fa;
for (int i = ; ( << i) <= d[x]; ++i) fa[i][x] = fa[i - ][fa[i - ][x]];
for (auto i : g[x]) {
dfs1(i, x);
siz[x] += siz[i];
if (siz[i] > siz[big[x]])
big[x] = i;
}
}
void dfs2(int x) {//dfs序给定后,后面的遍历全部倒着来
dfn[x] = ++cok;
if (big[x])
dfs2(big[x]);
for (int i = g[x].size() - ; i >= ; --i)
if (g[x][i] != big[x])
dfs2(g[x][i]);
}
void draw(int x, int k) {
c[d[x]] += k;
for (auto i : g[x])
if (!vis[i])
draw(i, k);
}
void dfs3(int x, bool is) {
for (auto i : g[x])
if (i != big[x])
dfs3(i, );
if (big[x])
dfs3(big[x], ), vis[big[x]] = ;
draw(x, );
while (R && a[R].x == x) ans[a[R].id] = c[a[R].k] - , --R;
if (big[x])
vis[big[x]] = ;
if (!is)
draw(x, -);
}
int find(int x, int k) {//倍增找k-祖先
for (int i = ; k && i >= ; --i)
if (k >= ( << i))
x = fa[i][x], k -= << i;
return x;
}
int main() {
read(n);
for (int i = , u; i <= n; ++i) read(u), g[u].push_back(i);
read(Q);
R = Q;
for (int i = ; i <= Q; ++i) read(a[i].x), read(a[i].k), a[i].id = i;
for (auto i : g[]) dfs1(i, ), _S[++tp] = i;
for (auto i : g[]) dfs2(i);
for (int i = , q; i <= Q; ++i) q = a[i].x, a[i].x = find(q, a[i].k), a[i].k = d[q];//询问转化
sort(a + , a + Q + , cmp);
for (int i = tp; i; --i) dfs3(_S[i], );//倒着dfs
for (int i = ; i <= Q; ++i) printf("%d ", ans[i]);
return ;
}

CF208E Blood Cousins的更多相关文章

  1. CF208E Blood Cousins 题解

    一个奇奇怪怪的复杂度很垃圾的线段树合并解法 通过分析可以发现,要找$x$的$k$辈兄弟,只需要找到$x$的$k$辈祖先,然后查找以该祖先为根的子树中和$x$深度相同的节点个数$-1$即可.也就是说,询 ...

  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. vue报错TypeError: Cannot read property 'protocol' of undefined

    错误信息如下所示: isURLSameOrigin.js?3934:57 Uncaught (in promise) TypeError: Cannot read property 'protocol ...

  2. whetstone

    https://www.cnblogs.com/findumars/p/4173040.html 下载源码:http://www.netlib.org/benchmark/whetstone.c ar ...

  3. thinkphp数据库连接

    https://www.kancloud.cn/manual/thinkphp5/118059 一.配置文件定义 常用的配置方式是在应用目录或者模块目录下面的database.php中添加下面的配置参 ...

  4. Linux和VMware

    1.1   Linux操作系统简介 是一个基于POSIX和UNIX的多用户.多任务.支持多线程和多CPU的操作系统.它能运行主要的UNIX工具软件.应用程序和网络协议.它支持32位和64位硬件.Lin ...

  5. macaca搭建

    对于新鲜的事务总是那么好奇,在自动化的过程中,有幸了解到macaca,记录下安装过程,具体介绍请移步官网:https://github.com/macacajs/ python版本参考:https:/ ...

  6. 向Hive中导入数据的方式

    一.Hive客户端:根据数据源不同划分 1.从本地文件系统中导入数据到hive表中: load data local inpath "path" [OVERWRITE] into ...

  7. @ContrllerAdvice全局异常

    @ControllerAdvice,是Spring3.2提供的新注解,它是一个Controller增强器,可对controller中被 @RequestMapping注解的方法加一些逻辑处理.最常用的 ...

  8. 阿里云E-HPC联合安世亚太、联科集团共建云超算生态

    5月23日,2018云栖大会武汉峰会,阿里云高级技术专家刘峥和张维,对弹性计算最新上线的 serverless (无服务器化)计算技术Bazaar及基于该技术的容器服务产品 Severless Kub ...

  9. spfa求次短路

    思路:先算出每个点到1的最短路d1[i],记录下路径,然后枚举最短路上的边 删掉之后再求一遍最短路,那么这时的最短路就可能是答案. 但是这个做法是错误的,可以被卡掉. 比如根据下面的例题生成的一个数据 ...

  10. 四-1、Cadence Allegro推荐操作方式和视图命令

    第四章:实用命令详解 1.Cadence Allegro推荐操作方式: 激活命令 选择操作对象的属性 设置相关的命令参数 单击对应的对象 结束命令 2.视图命令: