bzoj2588 Count on a tree
题意:给定一棵树,有点权,不带修改,询问路径点权第K大,强制在线。
这道题建主席树的方法好机智。按照BFS/DFS序建树,对于每个点,建出“这个点到根节点的路径上的点”组成的权值线段树,某个节点的树由父节点的树更改一条链得来。查询时用路径两个端点到根的线段树减去lca到根节点的线段树的2倍就得到了这条路径。注意lca的点权要特殊处理一下,不要把lca多减一次。据说只要分别减去lca和lca的父亲即可,查询的时候传4棵线段树的节点。
于是抖机灵强行传3个节点,单独处理lca,结果RE了一坨…..因为lca的权值小于区间中点的权值并不一定是lca处在[l,r]区间的左半部分,还有可能是lca在[l,r]区间的左边,加句rk[lca]>=l就过了.调了一小时,虚死….
#include<cstdio>
#include<algorithm>
using namespace std;
const int maxn=;
int n,m;
struct edge{
int to,next;
}lst[maxn<<];int len=;
int first[maxn];
void addedge(int a,int b){
lst[len].to=b;
lst[len].next=first[a];
first[a]=len++;
}
int c[maxn];
int q[maxn];
int depth[maxn],p[maxn][];
void bfs(){
int head=,tail=;
depth[]=;
q[tail++]=;
int x;
while(head!=tail){
x=q[head++];
for(int pt=first[x];pt;pt=lst[pt].next){
if(lst[pt].to==p[x][])continue;
p[lst[pt].to][]=x;
depth[lst[pt].to]=depth[x]+;
q[tail++]=lst[pt].to;
}
for(int j=;p[x][j];++j)p[x][j+]=p[p[x][j]][j];
}
}
int LCA(int u,int v){
if(depth[u]<depth[v])swap(u,v);
for(int j=;j>=;--j){
if(depth[p[u][j]]>=depth[v])u=p[u][j];
}
if(u==v)return u;
for(int j=;j>=;--j){
if(p[u][j]!=p[v][j]){
u=p[u][j];v=p[v][j];
}
}
return p[u][];
}
struct node{
int sum;
node *lch,*rch;
node(){
lch=rch=;
sum=;
}
}t[maxn*];int cnt=;
int tot=;
node *root[maxn];
int qx;
int seq[maxn],dict[maxn],rk[maxn];
void Insert(node *rt1,node* &rt2,int l,int r){
++cnt;rt2=t+cnt;
if(l==r){
rt2->sum=rt1->sum+;
rt2->lch=rt2->rch=t+;
return;
}
int mid=(l+r)>>;
if(qx<=mid){
rt2->rch=rt1->rch;
Insert(rt1->lch,rt2->lch,l,mid);
}else{
rt2->lch=rt1->lch;
Insert(rt1->rch,rt2->rch,mid+,r);
}
rt2->sum=rt2->lch->sum+rt2->rch->sum;
}
void build_all(){
root[]=t+;
root[]->rch=root[]->lch=t+;
root[]->sum=;
for(int i=;i<n;++i){
qx=rk[q[i]];
Insert(root[p[q[i]][]],root[q[i]],,tot);
}
}
bool cmp(const int &x,const int &y){
return c[x]<c[y];
}
int lca;
int query(node *rt1,node *rt2,node *rt3,int l,int r){
if(l==r)return l;
int mid=(l+r)>>;
int lsum=rt1->lch->sum+rt2->lch->sum-*(rt3->lch->sum);
if(l<=rk[lca]&&rk[lca]<=mid)lsum++;
//if(rk[lca]<=mid) 错误的写法.这时rk[lca]可以小于l
if(qx<=lsum)return query(rt1->lch,rt2->lch,rt3->lch,l,mid);
else{
qx-=lsum;
return query(rt1->rch,rt2->rch,rt3->rch,mid+,r);
}
}
int main(){
scanf("%d%d",&n,&m);
for(int i=;i<=n;++i){
scanf("%d",c+i);
seq[i]=i;
}
sort(seq+,seq+n+,cmp);
int old=c[seq[]]-;
for(int i=;i<=n;++i){
if(old!=c[seq[i]]){
old=c[seq[i]];++tot;
dict[tot]=c[seq[i]];
}
rk[seq[i]]=tot;
}
int a,b,k;
for(int i=;i<n;++i){
scanf("%d%d",&a,&b);
addedge(a,b);addedge(b,a);
}
bfs();
build_all();
int lastans=;
while(m--){
scanf("%d%d%d",&a,&b,&k);
a^=lastans;
qx=k;lca=LCA(a,b);
lastans=dict[query(root[a],root[b],root[lca],,tot)];
if(m)printf("%d\n",lastans);
else printf("%d",lastans);
}
return ;
}
bzoj2588 Count on a tree的更多相关文章
- 洛谷P2633/bzoj2588 Count on a tree (主席树)
洛谷P2633/bzoj2588 Count on a tree 题目描述 给定一棵N个节点的树,每个点有一个权值,对于M个询问(u,v,k),你需要回答u xor lastans和v这两个节点间第K ...
- BZOJ2588 Count on a tree 【树上主席树】
2588: Spoj 10628. Count on a tree Time Limit: 12 Sec Memory Limit: 128 MB Submit: 7577 Solved: 185 ...
- 主席树+dfs SPOJ BZOJ2588 Count on a tree
这道题我由于智障错误导致一直错. 在树上建主席树,加上lca思想,很简单. #include<bits/stdc++.h> using namespace std; ; struct no ...
- [BZOJ2588]Count on a tree(LCA+主席树)
题面 给定一棵N个节点的树,每个点有一个权值,对于M个询问(u,v,k),你需要回答u xor lastans和v这两个节点间第K小的点权.其中lastans是上一个询问的答案,初始为0,即第一个询问 ...
- [bzoj2588][count on a tree] (主席树+lca)
Description 给定一棵N个节点的树,每个点有一个权值,对于M个询问(u,v,k),你需要回答u xor lastans和v这两个节点间第K小的点权.其中lastans是上一个询问的答案,初始 ...
- 【填坑向】spoj COT/bzoj2588 Count on a tree
这题是学主席树的时候就想写的,,, 但是当时没写(懒) 现在来填坑 = =日常调半天lca(考虑以后背板) 主席树还是蛮好写的,但是代码出现重复,不太好,导致调试的时候心里没底(虽然事实证明主席树部分 ...
- BZOJ2588:Count on a tree(主席树)
Description 给定一棵N个节点的树,每个点有一个权值,对于M个询问(u,v,k),你需要回答u xor lastans和v这两个节点间第K小的点权.其中lastans是上一个询问的答案,初始 ...
- [Bzoj2588]Count on a tree(主席树+LCA)
Description 给定一棵N个节点的树,每个点有一个权值,对于M个询问(u,v,k),你需要回答u xor lastans和v这两个节点间第K小的点权.其中lastans是上一个询问的答案,初始 ...
- BZOJ2588 SPOJ10628 Count on a tree 【主席树】
BZOJ2588 Count on a tree 题目描述 给定一棵N个节点的树,每个点有一个权值,对于M个询问(u,v,k),你需要回答u xor lastans和v这两个节点间第K小的点权.其中l ...
随机推荐
- redis 学习笔记(4)-HA高可用方案Sentinel配置
上一节中介绍了master-slave模式,在最小配置:master.slave各一个节点的情况下,不管是master还是slave down掉一个,“完整的”读/写功能都将受影响,这在生产环境中显然 ...
- 浅谈设计模式--装饰者模式(Decorator Pattern)
挖了设计模式这个坑,得继续填上.继续设计模式之路.这次讨论的模式,是 装饰者模式(Decorator Pattern) 装饰者模式,有时也叫包装者(Wrapper),主要用于静态或动态地为一个特定的对 ...
- es6+移动轮播插件
前言:之前赶项目,都是直接用框架,对于touch事件是模拟两可,趁着有心情,用es6写一个原生移动轮播插件. 用了es6的新特性,确实挺爽的,说到es6,就不得不说到babel,博主已经码好了,直接用 ...
- opencv7-ml之svm
因为<opencv_tutorial>这部分只有两个例子,就先暂时介绍两个例子好了,在refman中ml板块有:统计模型.普通的贝叶斯分类器.KNN.SVM.决策树.boosting.随机 ...
- HTML5+JS 《五子飞》游戏实现(五)移动棋子
上一章 我们知道了怎么处理两个重要的吃棋动作,想要吃对方的棋子,首先得移动自己的棋子.现在里沃特跟大家分享分享,怎么移动棋子. 想要移动棋子,在页面上,首先要点击一下要移动的棋子,然后再点击一下目标位 ...
- Webpack配置示例和详细说明
/* * 请使用最新版本nodejs * 默认配置,是按生产环境的要求设置,也就是使用 webpack -p 命令就可以生成正式上线版本. * 也可以使用 webpack -d -w 命令,生成用于开 ...
- display~
- ubuntu14.04完全卸载mysql
1.删除 mysql1 sudo apt-get autoremove --purge mysql-server-5.0 2 sudo apt-get remove mysql-server 3 su ...
- 转载--web前端35个jQuery小技巧!
1. 禁止右键点击$(document).ready(function(){ $(document).bind("contextmenu",function(e){ ...
- C/C++中NULL的涵义
参考:百度知道NULL表示空指针,用于表示一个无效的指针,它的值为0(早期C语言的实现中可能有非0空指针,现在已经不用).对指针置NULL即标记指针无效,避免“野指针”的恶果.NULL在C/C++标准 ...