[luogu P2633] Count on a tree
[luogu P2633] Count on a tree
题目描述
给定一棵N个节点的树,每个点有一个权值,对于M个询问(u,v,k),你需要回答u xor lastans和v这两个节点间第K小的点权。其中lastans是上一个询问的答案,初始为0,即第一个询问的u是明文。
输入输出格式
输入格式:
第一行两个整数N,M。
第二行有N个整数,其中第i个整数表示点i的权值。
后面N-1行每行两个整数(x,y),表示点x到点y有一条边。
最后M行每行两个整数(u,v,k),表示一组询问。
输出格式:
M行,表示每个询问的答案。
输入输出样例
输入样例#1: 复制8 5 105 2 9 3 8 5 7 7 1 2 1 3 1 4 3 5 3 6 3 7 4 8 2 5 1 0 5 2 10 5 3 11 5 4 110 8 2输出样例#1: 复制2 8 9 105 7说明
HINT:
N,M<=100000
暴力自重。。。
来源:bzoj2588 Spoj10628.
本题数据为洛谷自造数据,使用CYaRon耗时5分钟完成数据制作。
主席树的变形,离散后,对于每个点u构造一棵新的主席树都建立在以fa[u][0]构造的主席树的基础之上。
其中,要按照dfs序来造,才能保证正确性。(用bfs序当然也可以咯)
然后询问两个点(x,y)时,根据主席树可加减的性质,只要在tree[x]+tree[y]-tree[lca(x,y)]-tree[fa[lca(x,y)]]这棵树上查询就可以了。
注意强制在线。
code:
#include <cstdio> #include <cstring> #include <algorithm> #include <tr1/unordered_map> #define ms(a,x) memset(a,x,sizeof a) using namespace std; void OJ() { #ifndef ONLINE_JUDGE freopen("in.txt","r",stdin); freopen("out.txt","w",stdout); #endif } namespace fastIO { #define gec(c) getchar(c) #define puc(c) putchar(c) char ch; inline int read() { ,f=; ch=getchar(); ') { if (ch=='-') f=-f; ch=gec(); } ') { x=(x<<)+(x<<)+ch-'; ch=getchar(); } return x*f; } ]; template <class T> inline void write(T x) { ) { puc('); return; } ) x=-x,puc('-'); ; x; x/=) nnn[++ttt]=x%; ); } inline void newline() { puc('\n'); } } using namespace fastIO; ; int n,m,cnt,b[N]; ],son[N<<]; ],dfn[N],rel[N],clo; tr1::unordered_map <int,int> ref; struct data { int x,i; } a[N]; inline bool cmp_x (const data &u,const data &v) { return u.x<v.x; } inline bool cmp_i (const data &u,const data &v) { return u.i<v.i; } struct node { node* l,* r; int v; node () { l=r=,v=; } } *r[N]; #define M ((l)+(r)>>1) inline void setup (node* &c,int l,int r) { c=new node(); if (l==r) return; setup(c->l,l,M),setup(c->r,M+,r); } inline void insert (node* &c,int l,int r,int x,node* his) { c=new node(); c->l=his->l,c->r=his->r,c->v=his->v+; if (l==r) return; if (x<=M) insert(c->l,l,M,x,his->l); ,r,x,his->r); } inline ) { if (l==r) return b[l]; v+=x->l->v,v+=y->l->v; v-=z->l->v,v-=o->l->v; if (k<=v) return reply(x->l,y->l,z->l,o->l,l,M,k); ,r,k-v); } void add (int x,int y) { nxt[++tot]=lnk[x],lnk[x]=tot,son[tot]=y; } void dfs (int x,int p) { dep[x]=dep[p]+,fa[x][]=p,rel[dfn[x]=++clo]=x; ; j<=; ++j) fa[x][j]=fa[fa[x][j-]][j-]; for (int j=lnk[x]; j; j=nxt[j]) { if (son[j]!=p) dfs(son[j],x); } } int lca (int x,int y) { if (dep[x]<dep[y]) swap(x,y); ) ; int dif=dep[x]-dep[y]; ; ~j; --j) <<j)) x=fa[x][j]; if (x==y) return x; ; ~j; --j) if (fa[x][j]!=fa[y][j]) x=fa[x][j],y=fa[y][j]; ]; } int main() { OJ(); ; n=read(),m=read(); ; i<=n; ++i) a[i].x=read(); ; i<n; ++i) { x=read(),y=read(); add(x,y),add(y,x); } dep[]=,dfn[]=,dfs(,); ; i<=n; ++i) a[i].i=dfn[i]; sort(a+,a++n,cmp_x); b[cnt=].x]=]=a[].x; ; i<=n; i++) ].x) b[ref[a[i].x]=++cnt]=a[i].x; sort(a+,a++n,cmp_i); setup(r[],,cnt); ; i<=n; ++i) insert(r[i],,cnt,]]]); ; i<=m; ++i) { x=read(),x^=lastans,y=read(),z=lca(x,y),o=fa[z][],k=read(); x=dfn[x],y=dfn[y],z=dfn[z],o=dfn[o]; write(lastans=reply(r[x],r[y],r[z],r[o],,cnt,k)); if (i^m) newline(); } ; }
[luogu P2633] Count on a tree的更多相关文章
- 主席树 || 可持久化线段树 || LCA || BZOJ 2588: Spoj 10628. Count on a tree || Luogu P2633 Count on a tree
		
题面: Count on a tree 题解: 主席树维护每个节点到根节点的权值出现次数,大体和主席树典型做法差不多,对于询问(X,Y),答案要计算ans(X)+ans(Y)-ans(LCA(X,Y) ...
 - 洛谷 P2633 Count on a tree
		
P2633 Count on a tree 题目描述 给定一棵N个节点的树,每个点有一个权值,对于M个询问(u,v,k),你需要回答u xor lastans和v这两个节点间第K小的点权.其中last ...
 - 洛谷P2633 Count on a tree(主席树上树)
		
题目描述 给定一棵N个节点的树,每个点有一个权值,对于M个询问(u,v,k),你需要回答u xor lastans和v这两个节点间第K小的点权.其中lastans是上一个询问的答案,初始为0,即第一个 ...
 - 洛谷 P2633 Count on a tree 主席树
		
在一棵树上,我们要求点 $(u,v)$ 之间路径的第$k$大数. 对于点 $i$ ,建立 $i$ 到根节点的一棵前缀主席树. 简单容斥后不难得出结果为$sumv[u]+sumv[v]−sumv[l ...
 - 洛谷 P2633 Count on a tree 题解
		
题面 对于每个点建立一颗主席树: 然后按照树上差分的思想统计主席树的前缀和: lca+主席树+前向星存表就可以了: #include <bits/stdc++.h> #define inc ...
 - 洛谷P2633 Count on a tree(主席树,倍增LCA)
		
洛谷题目传送门 题目大意 就是给你一棵树,每个点都有点权,每次任意询问两点间路径上点权第k小的值(强制在线). 思路分析 第k小......又是主席树了.但这次变成树了,无法直接维护前缀和. 又是树上 ...
 - ☆ [洛谷P2633] Count on a tree 「树上主席树」
		
题目类型:主席树+\(LCA\) 传送门:>Here< 题意:给出一棵树.每个节点有点权.问某一条路径上排名第\(K\)小的点权是多少 解题思路 类似区间第\(K\)小,但放在了树上. 考 ...
 - 洛谷P2633 Count on a tree 主席树
		
传送门:主席树 解题报告: 传送门! umm这题我还麻油开始做 所以 先瞎扯一波我的想法,如果错了我就当反面教材解释这种典型错误,对了我就不管了QwQ 就直接dfs,在dfs的过程中建树 然后就直接查 ...
 - P2633 Count on a tree
		
思路 运用树上差分的思想,转化成一个普通的主席树模型即可求解 代码 #include <cstdio> #include <algorithm> #include <cs ...
 
随机推荐
- vm中的nat模式不能连接到主机,以及不能上网
			
主机能够ping通vm中的ip,但是vm不能ping通主机ip vm不能ping通主机:主要是虚拟机中的gateway的设置和vm的网络编辑器中的网关id不一致造成的 vm不能上外网:dns没有配置正 ...
 - 2019 Lonsdor K518S VS K518ISE
			
2019 Lonsdor K518S VS K518ISE: The same: IMMO capabilities + Vehicle coverage. The difference: The u ...
 - 关于富文本编辑器—UEditor(java版)的使用,以及如何将UEditor的文件/图片上传路径改成绝对路径
			
突然发现好久没写博客了,感觉变懒了,是要让自己养成经常写文章的习惯才行.既可以分享自己的所学,和所想,和大家一起讨论,发现自己的不足的问题. 大家可能经常会用到富文本编辑器,今天我要说的是UEdito ...
 - DUILIB UI创建过程
			
函数调用过程: CDialogBuilder 内部过程循环创建控件树 上图中 在AttachDialog中设置窗口的主控件 并设置控件树的pm
 - matlab飞机飞行
			
function donghua4 %首先建立一个飞机模型 %然后写一个旋转仿射矩阵 %利用仿射变换改变飞机方向 clear;clc;TR=[1 50 41;1 51 50;2 51 1;3 51 2 ...
 - Linux环境下部署开源版“禅道”方法
			
1.开源版安装包下载(Linux系统版本查看命令 uname -a) 32位 [root@iZbp~]# wget http://dl.cnezsoft.com/zentao/9.0.1/ZenTao ...
 - js 手动插入meta标签和script标签
			
// 插入 meta 标签 var oMeta = document.createElement('meta'); oMeta.content = 'width=device-width, initi ...
 - 51行代码实现简单的PHP区块链
			
本文原始地址:php区块链demo 今年区块链特别火,我也很火啊.我火什么呢.前几年,公众平台出现,还得花时间去学去看,后来小程序出现,又得花时间精力去学去看.现在比特币.以太坊等去中心化货币带起了区 ...
 - springboot-yml内list、map组合写法
			
yml:myProps: varmaplist: key11: - t1 - t2 - t3 key22: - t11 - t22 - t33 list: - topic1 - topic2 - to ...
 - docker环境下的测试
			
docker作为容器常见用于快速部署,最近有个项目是基于docker的,总结一下docker的测试. 1.在主机中安装Docker: ubuntu安装:curl -s https://get.dock ...