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


通过分析可以发现,要找$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. PHP pos() 函数

    实例 输出数组中的当前元素的值: <?php$people = array("Peter", "Joe", "Glenn", &quo ...

  2. python数据可视化编程实战PDF高清电子书

    点击获取提取码:3l5m 内容简介 <Python数据可视化编程实战>是一本使用Python实现数据可视化编程的实战指南,介绍了如何使用Python最流行的库,通过60余种方法创建美观的数 ...

  3. 为什么需要将网站封装为app?

      网站封装为app是一种宝贵的资源,为客户提供稳定的平台,一个网站也是一个有效的工具,用于企业与其客户之间的通信.企业网站用户可以通过他们的笔记本电脑,台式机,平板电脑,智能手机以及带有浏览器设备的 ...

  4. C#-用Winform制作一个简单的密码管理工具

    为什么要做? 首先是为了练习一下c#. 想必大家都有过记不起某个平台的账号密码的经历,那种感受着实令人抓狂.那这么多账号密码根本记不住!我之前用python写过一个超级简单(连账号信息都写在代码里那种 ...

  5. Java Redis系列3(Jedis的使用+jedis连接池技术)

    Jedis的使用 什么是Jedis? 一款Java操作redis数据库的工具 使用步骤 1.下载redis所需的java包 2.使用步骤 import org.junit.Test; public c ...

  6. C#设计模式之2-抽象工厂模式

    抽象工厂模式(Abstract Factory Pattern) 该文章的最新版本已迁移至个人博客[比特飞],单击链接 https://www.byteflying.com/archives/391 ...

  7. Vuex mapGetter的基本使用

    getter相当于Vuex中的计算属性 对 state 做处理再返回 mapGetters 把 Store 中的 getters 映射到组件中的计算属性中 Store文件 import Vue fro ...

  8. golang 浮点型

    目录 前言 1.三要素 2.表现形式 3.类型 4.精度 5.格式化 6.使用细节 跳转 前言 不做文字的搬运工,多做灵感性记录 这是平时学习总结的地方,用做知识库 平时看到其他文章的相关知识,也会增 ...

  9. Spring——IOC(控制反转)与DI(依赖注入)

    IOC与DI的理解及使用 控制反转IOC(Inversion of Control)是一种设计思想,DI(依赖注入)是实现IOC的一种方法.在没有IOC的程序中,我们使用面向对象编程,对象的创建于对象 ...

  10. 阿里云日志服务SLS

    前言: 刚入职实习了几天,我发现我的任务就是学习阿里云日志服务这块业务内容,这个功能和mysql一样,但是速度和视觉却是甩mysql这类数据库几条街. 当得知公司没人会这项技术后(在这之前我也没听过, ...