hdu 5111 树链剖分加函数式线段树
这题说的是给了两棵树,各有100000 个节点,然后Q个操作Q<=50000; 每个操作L1 R1 L2 R2。因为对于每棵树都有一个与本棵树其他点与众不同的值, 最后问 在树上从L1到R1这条路径上与第二棵树L2 到 R2 这条路上的点 的权值相等的有多少个
这题挺麻烦的 写的想吐了
首先将第一棵树进行树剖,然后通过树剖可以离散出这颗树的每个点的编号从1,2,3,4...N1,然后将第二棵树进行树剖,按照树剖的值依次插入,以第一棵树离散出的值为叶节点的函数式线段树,
如果第二棵树的值在第一棵树种中不到,那么就直接将他插在值为0的叶节点上。我们可以知道在书剖中他们在同一条链上是相等的,所以他们在函数式线段树中也是连续的。通过这样我们可以使用类似区间第K大的解法解决这个问题
#include <iostream>
#include <cstdio>
#include <algorithm>
#include <string.h>
#include <vector>
#pragma comment(linker,"/STACk:10240000,10240000")
using namespace std;
const int maxn =;
vector<int> F[][maxn],rea[],fro[];
int p[][maxn],fp[][maxn],XX[],top[][maxn];
int depth[][maxn],num[][maxn],son[][maxn],fa[][maxn];
int L1[maxn],V[maxn],data[][maxn],T[maxn];
int Ls[maxn*],Rs[maxn*],Ve[maxn*],len;
struct point{
int x,loc;
bool operator<(const point &A)const{
return x<A.x||(x==A.x&&loc<A.loc);
}
}PP[maxn];
void inti(int loc, int n){
for(int i=; i<=n; ++i)
F[loc][i].clear();
}
void dfs(int loc, int now, int per,int dep){
fa[loc][now]=per;
son[loc][now]=-;
depth[loc][now]=dep;
num[loc][now]=;
int ge=F[loc][now].size();
for(int i=; i<ge; ++i){
int to = F[loc][now][i];
if(to==per||to==now)continue;
dfs(loc,to,now,dep+);
num[loc][now]+=num[loc][to];
if(son[loc][now]==-||num[loc][ son[loc][now] ]<num[loc][to]) son[loc][now]=to;
}
}
void fine(int loc, int now , int X,int per){
top[loc][now]=X;
XX[loc]++;
p[ loc ][ now ]=XX[ loc ];
fp[ loc ][ XX[loc] ]=now;
if( son[ loc ][ now ] != - )
fine( loc , son[loc][now] , X ,now);
int ge=F[loc][now].size();
for(int i=; i<ge; ++i){
int to=F[loc][now][i];
if( to==son[loc][now]||to==now||to==per ) continue;
fine(loc,to,to,now);
}
}
void inser(int L, int R, int K, int per, int &x){
x = ++len;
Ls[x]=Ls[per];
Rs[x]=Rs[per];
Ve[x]=Ve[per]+;
if(L==R)return ;
int mid=(L+R)>>;
if(K<=mid) inser( L, mid , K, Ls[per] , Ls[x] );
else inser( mid+ , R , K , Rs[per] , Rs[x] );
}
int ansed,Lc,Rc;
void query(int L, int R, int per, int cur){
if(Lc<=L&&R<=Rc){
ansed += Ve[cur]-Ve[per];return;
}
if(L>=R)return ;
int mid = (L+R)>>;
if(Lc<=mid) query(L, mid , Ls[per],Ls[cur]);
if(Rc>mid) query(mid+, R, Rs[per],Rs[cur]);
} void solve(int loc, int p1, int p2){
int f1=top[loc][p1],f2=top[loc][p2];
int num=;
while(f1!=f2){
num++;
if(num>) break;
if(depth[loc][f1]<depth[loc][f2]){
int t=p1; p1=p2; p2=t;
t=f1; f1=f2; f2=t;
}
fro[loc].push_back( p[loc][f1] );
rea[loc].push_back( p[loc][p1] );
p1=fa[loc][f1];
f1=top[loc][p1];
}
if(depth[loc][p1]<depth[loc][p2]){
int t=p1; p1=p2; p2=t;
}
fro[loc].push_back(p[loc][p2]);
rea[loc].push_back(p[loc][p1]);
}
int main()
{
int N1, N2;
while(scanf("%d",&N1)==){
for(int i=; i<N1; ++i){
int f;
scanf("%d",&f);
F[][f].push_back(i+);
}
XX[]=;
dfs(,,,);
fine(,,,);
for(int i=; i<N1; ++i){
scanf("%d",&data[][i]);
PP[i].x = data[][i];
PP[i].loc = i+;
}
sort(PP,PP+N1);
for(int i=; i<N1; ++i)
{
V[i]=PP[i].x;
L1[i]=p[ ][ PP[i].loc ];
} scanf("%d",&N2);
for(int i=; i<N2; ++i){
int fa;
scanf("%d",&fa);
F[][fa].push_back(i+);
}
XX[]=;
dfs(,,,);
fine(,,,);
for(int i=; i<=N2; ++i){
scanf("%d",&data[][i]);
int loc = lower_bound(V,V+N1,data[][i])-V;
if(V[loc]==data[][i]){
data[][i]=L1[loc];
}else data[][i]=;
}
Ls[]=Rs[]=Ve[]=len=;
for(int i=; i <= N2; ++ i ){
int loc=fp[ ][ i ];
inser(,N1,data[ ][ loc ],T[ i- ],T[ i ]);
}
int Q;
scanf("%d",&Q);
for(int i=; i<Q; ++i){
int L1,R1,L2,R2;
scanf("%d%d%d%d",&L1,&R1,&L2,&R2);
rea[].clear(); fro[].clear(); rea[].clear(); fro[].clear();
solve(,L1,R1);
solve(,L2,R2);
ansed=;
for(int j=; j<(int)rea[].size(); ++j){
Lc=fro[][j]; Rc=rea[][j];
if(Lc>Rc){ int t=Lc; Lc=Rc; Rc=t; }
for(int k=; k<(int)rea[].size(); ++k){
int L=fro[][k],R=rea[][k];
if(L>R){ int t=L; L=R; R=t; }
query( ,N1,T[L-],T[R] );
}
}
printf("%d\n",ansed);
}
inti(,N1);
inti(,N2);
}
return ;
}
hdu 5111 树链剖分加函数式线段树的更多相关文章
- BZOJ4012[HNOI2015]开店——树链剖分+可持久化线段树/动态点分治+vector
题目描述 风见幽香有一个好朋友叫八云紫,她们经常一起看星星看月亮从诗词歌赋谈到 人生哲学.最近她们灵机一动,打算在幻想乡开一家小店来做生意赚点钱.这样的 想法当然非常好啦,但是她们也发现她们面临着一个 ...
- BZOJ 2243: [SDOI2011]染色 树链剖分 倍增lca 线段树
2243: [SDOI2011]染色 Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeOnline/pr ...
- BZOJ 2243:染色(树链剖分+区间合并线段树)
[SDOI2011]染色Description给定一棵有n个节点的无根树和m个操作,操作有2类:1.将节点a到节点b路径上所有点都染成颜色c:2.询问节点a到节点b路径上的颜色段数量(连续相同颜色被认 ...
- HDU 5029 Relief grain 树链剖分打标记 线段树区间最大值
Relief grain Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://acm.hdu.edu.cn/showproblem.php?pid= ...
- HDU 3966 Aragorn's Story(模板题)【树链剖分】+【线段树】
<题目链接> 题目大意: 给定一颗带点权的树,进行两种操作,一是给定树上一段路径,对其上每个点的点权增加或者减少一个数,二是对某个编号点的点权进行查询. 解题分析: 树链剖分的模板题,还不 ...
- 焦作网络赛E-JiuYuanWantstoEat【树链剖分】【线段树】
You ye Jiu yuan is the daughter of the Great GOD Emancipator. And when she becomes an adult, she wil ...
- 【Codeforces】【网络流】【树链剖分】【线段树】ALT (CodeForces - 786E)
题意 现在有m个人,每一个人都特别喜欢狗.另外还有一棵n个节点的树. 现在每个人都想要从树上的某个节点走到另外一个节点,且满足要么这个人自带一条狗m,要么他经过的所有边h上都有一条狗. 2<=n ...
- LOJ2269 [SDOI2017] 切树游戏 【FWT】【动态DP】【树链剖分】【线段树】
题目分析: 好题.本来是一道好的非套路题,但是不凑巧的是当年有一位国家集训队员正好介绍了这个算法. 首先考虑静态的情况.这个的DP方程非常容易写出来. 接着可以注意到对于异或结果的计数可以看成一个FW ...
- HYSBZ 4034 【树链剖分】+【线段树 】
<题目链接> 题目大意: 有一棵点数为 N 的树,以点 1 为根,且树点有权值.然后有 M 个 操作,分为三种: 操作 1 :把某个节点 x 的点权增加 a . 操作 2 :把某个节点 x ...
随机推荐
- scala函数进阶篇
1.求值策略scala里有两种求值策略Call By Value -先对函数实参求值,在函数体中用这个求出的参数值.Call By Name -先不对函数实参求值,而是函数实参每次在函数体内被用到时都 ...
- Maven(一)-- 基础知识
一.Maven的基本概念 Maven(翻译为"专家","内行")是跨平台的项目管理工具.主要服务于基于Java平台的项目构建,依赖管理和项目信息管理. 1.项目 ...
- exp/imp与expdp/impdp区别
在平常备库和数据库迁移的时候,当遇到大的数据库的时候在用exp的时候往往是需要好几个小时,耗费大量时间.oracle10g以后可以用expdp来导出数据库花费的时间要远小于exp花费的时间,而且文件也 ...
- poj_1475 BFS+BFS
题目大意 推箱子游戏的基本玩法,求将箱子推到目的地的推动次数最少(并不是人移动总次数)的人移动路径. 题目分析 求最短路径的搜索问题,使用BFS.注意题目求的是 推动次数最少,因此将箱子移动作为状态, ...
- 【TP3.2】 动态切换数据库方法
1 config 配置: 'connection' => 'mysql://root:root@localhost:3306/dbname', connection 数据库连接字符串,后面代码 ...
- java基础---->FilenameFilter之文件过滤
FilenameFilter用于对列表中文件名的过滤,今天我们就开始java中FilenameFilter的学习.好多年了,你一直在我的伤口中幽居,我放下过天地,却从未放下过你,我生命中的千山万水,任 ...
- 微软 IIS HTTP.sys漏洞原理学习以及POC
零.MS15-034POC核心部分(参考巡风): socket.setdefaulttimeout(timeout) s = socket.socket(socket.AF_INET, socket. ...
- Python的反序列化漏洞
最近准备研究一下反序列化漏洞,但是Java代码看不懂,所以先找一个python的看起,毕竟这种漏洞在python.php中一样存在,等研究明白了基本原理后去看java的反序列化漏洞. python反序 ...
- 从零打造在线网盘系统之Hibernate框架起步
欢迎浏览Java工程师SSH教程从零打造在线网盘系统系列教程,本系列教程将会使用SSH(Struts2+Spring+Hibernate)打造一个在线网盘系统,本系列教程是从零开始,所以会详细以及着重 ...
- Codeforces Round #438 by Sberbank and Barcelona Bootcamp (Div. 1 + Div. 2 combined)
A. Bark to Unlock 题目链接:http://codeforces.com/contest/868/problem/A 题目意思:密码是两个字符组成的,现在你有n个由两个字符组成的字符串 ...