题目链接:

http://codeforces.com/gym/101161/attachments

题意:

给出节点数为$n$的树

有$q$次询问,输出$a$节点到$b$节点路程中,经过的边的中位数

数据范围:

$1\leq n \leq 100000$

$1\leq q \leq 100000$

分析:

建一颗主席树,不同的是,节点的根继承的是父节点的根

再用$lca$求出公共祖先$tree[a]+tree[b]-2*tree[lca(a,b)]$

ac代码:

#include<bits/stdc++.h>
using namespace std;
#define ll long long
#define pii pair<int,int>
const int maxn = 5e4+100;
struct Node{
int L,R,num;
}tree[maxn*20];
int dep[maxn],fa[maxn][20],n,cnt;
int root[maxn],vis[maxn];
vector<pii>ve[maxn]; void update(int x,int&root,int st,int en){
tree[++cnt]=tree[root];
root=cnt;
tree[root].num++;
if(st==en)return ;
int md=(st+en)/2;
if(x<=md)
update(x,tree[root].L,st,md);
else
update(x,tree[root].R,md+1,en);
} void dfs(int x,int fx,int w){
if(vis[x])return ;
vis[x]=1;
root[x]=root[fx];
if(x!=1)update(w,root[x],1,1e5);
dep[x]=dep[fx]+1;
fa[x][0]=fx;
for(int i=1;(1<<i)<dep[x];i++)
fa[x][i]=fa[fa[x][i-1]][i-1];
for(int i=0;i<ve[x].size();i++)
dfs(ve[x][i].first,x,ve[x][i].second);
}
int lca(int a,int b){
if(dep[a]>dep[b])swap(a,b);
for(int i=19;i>=0;i--)
if(dep[fa[b][i]]>=dep[a])b=fa[b][i];
if(a==b)return a;
for(int i=19;i>=0;i--)
if(fa[a][i]!=fa[b][i])a=fa[a][i],b=fa[b][i];
return fa[a][0];
} int quer(int a,int b,int c,int st,int en,int k){
if(st==en)return st;
int dd=tree[tree[a].L].num+tree[tree[b].L].num-2*tree[tree[c].L].num;
//cout<<dd<<" "<<k<<endl;
int md=(st+en)/2;
if(dd>=k)return quer(tree[a].L,tree[b].L,tree[c].L,st,md,k);
return quer(tree[a].R,tree[b].R,tree[c].R,md+1,en,k-dd);
}
int main()
{
int T,q;
scanf("%d",&T);
while(T--){
scanf("%d",&n);
cnt=0;
for(int i=1;i<=n;i++){
ve[i].clear();
vis[i]=0;
for(int j=0;j<20;j++)
fa[i][j]=0;
}
for(int i=1;i<=n-1;i++){
int a,b,c;
scanf("%d %d %d",&a,&b,&c);
ve[a].push_back(make_pair(b,c));
ve[b].push_back(make_pair(a,c));
} dfs(1,0,-1);
//cout<<fa[6][2]<<endl;
scanf("%d",&q);
while(q--){
int a,b,c;
scanf("%d %d",&a,&b);
c=lca(a,b);
int len=dep[a]+dep[b]-2*dep[c];
//cout<<c<<" "<<len<<endl;
if(len%2==0){
int ans1=quer(root[a],root[b],root[c],1,1e5,len/2);
int ans2=quer(root[a],root[b],root[c],1,1e5,len/2+1);
printf("%d",(ans1+ans2)/2);
if((ans1+ans2)%2)printf(".5");
else printf(".0");
}
else
printf("%d.0",quer(root[a],root[b],root[c],1,1e5,len/2+1));
printf("\n");
}
}
return 0;
}

  

codeforces gym #101161E - ACM Tax(lca+主席树)的更多相关文章

  1. Codeforces Gym101161E:ACM Tax(主席树+LCA)

    题目链接 题意 给出一棵有边权的树,然后给出q个查询,每次查询问两个结点的路径上的边的长度的中位数是多少. 思路 这道题目是用主席树(用权值当结点)和LCA来做的. 和之前做过的区间第K大类似,这道题 ...

  2. UVALive - 7831 :ACM Tax (主席树求树路径上中位数:LCA+主席树)

    题意:给定一棵带权树,Q次询问,每次询问路径上的中位数. 思路:中位数分边数奇偶考虑,当当边数为num=奇时,结果就算路径第num/2+1大,用主席树做即可... (做了几道比较难的主席树,都wa了. ...

  3. Gym101161:ACM Tax (主席树)(占位)

    题意:给定一个带权树,Q次询问,每次回答某简单路径上的权值中位数. 思路:记录根到节点的主席树,主席树可以找到路径的第K大权值.(记住,这里是可以不用二分的,不要想多了.) 奇数条边直接找中位数,偶数 ...

  4. LCA+主席树 (求树上路径点权第k大)

      SPOJ 10628. Count on a tree (树上第k大,LCA+主席树) 10628. Count on a tree Problem code: COT You are given ...

  5. 【bzoj3123】[Sdoi2013]森林 倍增LCA+主席树+启发式合并

    题目描述 输入 第一行包含一个正整数testcase,表示当前测试数据的测试点编号.保证1≤testcase≤20. 第二行包含三个整数N,M,T,分别表示节点数.初始边数.操作数.第三行包含N个非负 ...

  6. SPOJ 10628 Count on a tree(Tarjan离线LCA+主席树求树上第K小)

    COT - Count on a tree #tree You are given a tree with N nodes.The tree nodes are numbered from 1 to  ...

  7. 【SPOJ】10628. Count on a tree(lca+主席树+dfs序)

    http://www.spoj.com/problems/COT/ (速度很快,排到了rank6) 这题让我明白了人生T_T 我知道我为什么那么sb了. 调试一早上都在想人生. 唉. 太弱. 太弱. ...

  8. BZOJ2588Count on a tree——LCA+主席树

    题目描述 给定一棵N个节点的树,每个点有一个权值,对于M个询问(u,v,k),你需要回答u xor lastans和v这两个节点间第K小的点权.其中lastans是上一个询问的答案,初始为0,即第一个 ...

  9. spoj COT - Count on a tree (树上第K小 LCA+主席树)

    链接: https://www.spoj.com/problems/COT/en/ 思路: 首先看到求两点之前的第k小很容易想到用主席树去写,但是主席树处理的是线性结构,而这道题要求的是树形结构,我们 ...

随机推荐

  1. Angular 惰性路由

    根路由上的一个 loadChildren 属性,设置为一个字符串.这样就是惰性路由了. angular6 这样写:loadChildren: './background-check/backgroun ...

  2. linux gcc安装

    2004年4月20日最新版本的GCC编译器3.4.0发布了.目前,GCC可以用来编译C/C++.FORTRAN.java.OBJC.ADA等语言的程序,可根据需要选择安装支持的语言.GCC 3.4.0 ...

  3. 安装Nvida 显示环境

    查看是否能正确加载nvidia 驱动 在终端输入 (glxinfo 需要安装mesa-utils) 如果可以正确加载了nvidia驱动 那么在输入的内容中可以看到NVIDIA 字样 如果GPU是Int ...

  4. C++标准库里自带的数值类型和字符串互相转换函数

    需要包含头文件 #include <string> 数值类型转成string类型: string to_string(int val); string to_string(unsigned ...

  5. javascript--HTML DOM常用元素对象

    二,Select:访问select元素 属性:.selectedIndex  获取select中当前选中项的下标 .options  获取select中所有的option元素 返回值为数组 .opti ...

  6. python2.7.5安装docker-compose的方法

    yum -y install epel-release && yum install -y python-pip && pip install --upgrade pi ...

  7. vue,onerror实现当图片加载失败时使用默认图

    1. 2.

  8. 关于Objective C的私有函数

    (1)很多从其他语言(例如C++)转到objective c的初学者,往往会问到一个问题,如何定义类的私有函数?这里的“私有函数”指的是,某个函数只能在类的内部使用,不能在类的外部,或者派生类内部使用 ...

  9. zlog日志函数库

    在C的世界里面没有特别好的日志函数库(就像JAVA里面的的log4j,或者C++的log4cxx).C程序员都喜欢用自己的轮子.printf就是个挺好的轮子,但没办法通过配置改变日志的格式或者输出文件 ...

  10. 如何让某些用户对Marketing Cloud的contact数据只能实施只读操作

    打开maintain business role这个应用: 创建一个新的business role,然后添加下列这几个catalogs: SAP_CEC_BC_MKT_ADM_PC Marketing ...