[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的更多相关文章

  1. 主席树 || 可持久化线段树 || 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) ...

  2. 洛谷 P2633 Count on a tree

    P2633 Count on a tree 题目描述 给定一棵N个节点的树,每个点有一个权值,对于M个询问(u,v,k),你需要回答u xor lastans和v这两个节点间第K小的点权.其中last ...

  3. 洛谷P2633 Count on a tree(主席树上树)

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

  4. 洛谷 P2633 Count on a tree 主席树

    在一棵树上,我们要求点 $(u,v)$ 之间路径的第$k$大数. 对于点 $i$  ,建立 $i$  到根节点的一棵前缀主席树. 简单容斥后不难得出结果为$sumv[u]+sumv[v]−sumv[l ...

  5. 洛谷 P2633 Count on a tree 题解

    题面 对于每个点建立一颗主席树: 然后按照树上差分的思想统计主席树的前缀和: lca+主席树+前向星存表就可以了: #include <bits/stdc++.h> #define inc ...

  6. 洛谷P2633 Count on a tree(主席树,倍增LCA)

    洛谷题目传送门 题目大意 就是给你一棵树,每个点都有点权,每次任意询问两点间路径上点权第k小的值(强制在线). 思路分析 第k小......又是主席树了.但这次变成树了,无法直接维护前缀和. 又是树上 ...

  7. ☆ [洛谷P2633] Count on a tree 「树上主席树」

    题目类型:主席树+\(LCA\) 传送门:>Here< 题意:给出一棵树.每个节点有点权.问某一条路径上排名第\(K\)小的点权是多少 解题思路 类似区间第\(K\)小,但放在了树上. 考 ...

  8. 洛谷P2633 Count on a tree 主席树

    传送门:主席树 解题报告: 传送门! umm这题我还麻油开始做 所以 先瞎扯一波我的想法,如果错了我就当反面教材解释这种典型错误,对了我就不管了QwQ 就直接dfs,在dfs的过程中建树 然后就直接查 ...

  9. P2633 Count on a tree

    思路 运用树上差分的思想,转化成一个普通的主席树模型即可求解 代码 #include <cstdio> #include <algorithm> #include <cs ...

随机推荐

  1. [20180603]Python读写csv

    原文:https://docs.python.org/3/library/csv.html 进入python官网后,选择document,然后选择library reference. 读取: impo ...

  2. css学习-》养成路线

    雅虎工程师提供的CSS初始化示例代码 body,div,dl,dt,dd,ul,ol,li,h1,h2,h3,h4,h5,h6,pre,code,form,fieldset,legend,input, ...

  3. C/C++中数据的存储

    学java时了解到不同的数据在系统中存储的位置不一样,有的存在栈里,有的存在堆里.学C/C++时没注意过这个,最近学数据结构时遇到了问题:在定义一个结构体的指针时,系统如何给它分配的空间?从而让我想去 ...

  4. MyBatis笔记(二) 最简单的insert命令

    接上一篇随笔.这里没有用到MyBatis最关键的映射器接口,因此只做个简单的insert操作,update和delete同理,就不再赘述了. 直接上代码: 首先是dao包下的UserDAO.java文 ...

  5. mongo 修改器

    [$inc] 作用:修改器$inc可以对文档的某个值为数字型(只能为满足要求的数字)的键进行增减的操作. Example: db.b.update({"uid" : "2 ...

  6. Flutter进阶—点击、拖动和其他手势

    Flutter中的手势系统有两个层次.第一层具有原始指针事件,其描述了穿过屏幕的指针(例如触摸.鼠标和触控笔)的位置和移动.第二层具有手势,其描述由一个或多个指针移动组成的语义动作. 指针指针代表用户 ...

  7. log4J日志框架

    log4j的配置:log4j是一个日志输出框架,就是用于输出日志的,主流框架大部分都是Log4j输出.Spring框架也可以通过Log4j输出日志 Log4j提供了强大的日志输出的自定义功能(1)通过 ...

  8. uirecorder 启动webdriver服务报错

    在安装好uirecorder后,执行起来是各种错误. 不是少这个就是缺那个,也是因为自己对自动化测试知识太匮乏. 导致刚开始走自动化测试绕了很多弯路,报个错都不知所措.后来才知道要多看ERROR后面的 ...

  9. AutoCAD2015有时候会显示乱七八糟的线

    问题描述:AutoCAD2015以上版本有时候打开一张图,会出现乱七八糟的线 解决方案: 这是由于硬件加速平滑线显示引起的,可以如下修改

  10. python经典程序

    输入输出 #判断输入整数是否在[0,100]之间 num = eval(input("请输入一个整数:")) if num > 100 or num < 0: #判断[ ...