题目描述

现在有一颗以 1 为根节点的由 n 个节点组成的树,树上每个节点上都有一个权值 \(v_i\)。现在有 Q 次操作,操作如下:

  • 1 x y :查询节点 x 的子树中与 y 异或结果的最大值。
  • 2 x y z :查询路径 x到 y 上点与 z 异或结果最大值


输入格式

第一行是两个数字 n , Q 。

第二行是 n 个数字用空格隔开,第 i 个数字 \(v_i\) 表示点 i 上的权值。

接下来 n−1 行,每行两个数, x,y ,表示节点 x 与 y 之间有边。

接下来 Q 行,每一行为一个查询,格式如上所述。


输出格式

对于每一个查询,输出一行,表示满足条件的最大值。


样例

样例输入

7 5
1 3 5 7 9 2 4
1 2
1 3
2 4
2 5
3 6
3 7
1 3 5
2 4 6 3
1 5 5
2 5 7 2
1 1 9

样例输出

7
6
12
11
14



#### 数据范围与提示

对于 10% 的数据,有 1≤n,Q≤100 。

对于 20% 的数据,有 1≤n,Q≤1000。

对于 40% 的数据,有 1≤n,Q≤10000。

对于 100% 的数据,有 1≤n,Q≤100000。

查询 1 中的 y≤ \(2^{30}\),查询中的 z≤ \(2^{30}\)。


Solution

这道题,就是一道可持久化Trie树

关于可持久化Trie树,其实也很简单,有以下前置技能:

  • 主席树
  • 01 \(Trie\)树

1.储存

使用主席树的储存结构来储存01 \(Trie\) 树。

建立 \(n\) 个虚点作为根,然后记录每个点的左儿子右儿子以及计数。

2.查询

关于查询,其实和带修改的 \(Trie\) 树 差不多,当且仅当已经构建的新树中该节点的 \(num\) 值大于0,我们才可以继续下去查询.最后输出最大值即可。

然后最后面套个剖分即可。


代码

#include<bits/stdc++.h>
#define ll long long
using namespace std;
const int maxn=200008;
ll c[maxn],cc[maxn];
int son[maxn],n,q;
int top[maxn],siz[maxn];
int dep[maxn],fa[maxn]; struct sj{
int to;
int next;
}a[maxn];
int size,head[maxn]; void add(int x,int y)
{
a[++size].to=y;
a[size].next=head[x];
head[x]=size;
} ll ch[maxn*40][2];
ll tot,T[maxn];
ll num[maxn*40]; int insert(int pre,ll x,int v)
{
int u=++tot;
ll c=((x>>v)&1);
ch[u][0]=ch[pre][0];
ch[u][1]=ch[pre][1];
num[u]=num[pre]+1;
if(v>=0)
ch[u][c]=insert(ch[pre][c],x,v-1);
return u;
} ll ans,now[2];
int query(int l,int r,ll x,int v)
{
ll c=((x>>v)&1);
now[0]=num[ch[r][0]]-num[ch[l][0]];
now[1]=num[ch[r][1]]-num[ch[l][1]];
if(now[c^1])
{
ans+=(1<<v);
if(v>=0)
query(ch[l][c^1],ch[r][c^1],x,v-1);
}
else
if(v>=0)
query(ch[l][c],ch[r][c],x,v-1);
} void dfs(int x)
{
siz[x]=1;
for(int i=head[x];i;i=a[i].next)
{
int tt=a[i].to;
if(!siz[tt])
{
dep[tt]=dep[x]+1;
fa[tt]=x;
dfs(tt);
siz[x]+=siz[tt];
if(siz[tt]>siz[son[x]])
son[x]=tt;
}
}
} int id[maxn],dum;
void dfs1(int x,int y)
{
top[x]=y;
id[x]=++dum;
c[dum]=cc[x];
if(son[x])
dfs1(son[x],y);
for(int i=head[x];i;i=a[i].next)
{
int tt=a[i].to;
if(!top[tt])
if(tt!=son[x])
dfs1(tt,tt);
}
} int check(int x,int y,int w)
{
ll rest=0;
while(top[x]!=top[y])
{
if(dep[top[x]]<dep[top[y]])
swap(x,y);
ans=0;
query(T[id[top[x]]-1],T[id[x]],w,32);
rest=max(ans,rest);
x=fa[top[x]];
}
if(id[x]>id[y])
swap(x,y);
ans=0;
query(T[id[x]-1],T[id[y]],w,32);
rest=max(rest,ans);
return rest;
} int main()
{
scanf("%d%d",&n,&q);
for(int i=1;i<=n;i++)
scanf("%lld",&cc[i]);
for(int i=1;i<n;i++)
{
int x,y;
scanf("%d%d",&x,&y);
add(x,y);
add(y,x);
}
dfs(1); dfs1(1,1);
for(int i=1;i<=n;i++)
T[i]=insert(T[i-1],c[i],32);
while(q--)
{
int opt,x,y,z;
scanf("%d",&opt);
if(opt==1)
{
scanf("%d%d",&x,&y);
ans=0;
query(T[id[x]-1],T[id[x]+siz[x]-1],y,32);
cout<<ans<<endl;
}
else
{
scanf("%d%d%d",&x,&y,&z);
if(x!=y)
cout<<check(x,y,z)<<endl;
else
cout<<(cc[x]^z)<<endl;
}
}
}

[TJOI2018] Xor 异或 (可持久化Trie,树链剖分)的更多相关文章

  1. P2420 让我们异或吧 (树链剖分,异或前缀和)

    题目描述 异或是一种神奇的运算,大部分人把它总结成不进位加法. 在生活中-xor运算也很常见.比如,对于一个问题的回答,是为1,否为0.那么: (A是否是男生 )xor( B是否是男生)=A和B是否能 ...

  2. 2019 ACM-ICPC 西安全国邀请赛 E-Tree 树链剖分+线段树

    题意 给一颗带点权的树,三种操作 \(1~s~t\) 修改从1到s的路径上的所有点,\(a[i]=a[i]|t\) \(2~s~t\) 修改从1到s的路径上的所有点,\(a[i]=a[i]\& ...

  3. BZOJ5338 [TJOI2018] Xor 【可持久化Trie树】【dfs序】

    题目分析: 很无聊的一道题目.首先区间内单点对应异或值的询问容易想到trie树.由于题目在树上进行,case1将路径分成两段,然后dfs的时候顺便可持久化trie树做询问.case2维护dfs序,对d ...

  4. 洛谷P4592 [TJOI2018]异或 【可持久化trie树】

    题目链接 BZOJ4592 题解 可持久化trie树裸题 写完就A了 #include<algorithm> #include<iostream> #include<cs ...

  5. 51nod 1295 XOR key (可持久化Trie树)

    1295 XOR key  题目来源: HackerRank 基准时间限制:1.5 秒 空间限制:262144 KB 分值: 160 难度:6级算法题   给出一个长度为N的正整数数组A,再给出Q个查 ...

  6. 51nod 1295 XOR key | 可持久化Trie树

    51nod 1295 XOR key 这也是很久以前就想做的一道板子题了--学了一点可持久化之后我终于会做这道题了! 给出一个长度为N的正整数数组A,再给出Q个查询,每个查询包括3个数,L, R, X ...

  7. bzoj 3261: 最大异或和 (可持久化trie树)

    3261: 最大异或和 Time Limit: 10 Sec  Memory Limit: 512 MB Description       给定一个非负整数序列 {a},初始长度为 N.       ...

  8. BZOJ3261: 最大异或和(可持久化trie树)

    题意 题目链接 Sol 设\(sum[i]\)表示\(1 - i\)的异或和 首先把每个询问的\(x \oplus sum[n]\)就变成了询问前缀最大值 可持久化Trie树维护前缀xor,建树的时候 ...

  9. 【bzoj3261】【最大异或和】可持久化trie树+贪心

    [pixiv] https://www.pixiv.net/member_illust.php?mode=medium&illust_id=61705397 Description 给定一个非 ...

随机推荐

  1. dp cf 1700 最近几天的刷题

    C. Number of Ways 这个题目的意思是,把这个n的序列分成三个连续的部分,要求这三个部分的和是一样的.问这种划分的方法有多少种. 这个题目和之前写过的数字划分有点像,这个就是要先进行前缀 ...

  2. Ubuntu 忘记root user密码 关闭图形界面

    忘记root密码 删除recovery nomodeset 才删除的后面添加 quiet splash rw init=/bin/bash.然后按F10, 启动 忘记User密码 http://blo ...

  3. js判断是否为app

    var ua = navigator.userAgent; var isapp = ua.match("lenovomallapp") == null ? 0 : 1;

  4. 二叉树、二叉搜索树、平衡二叉树、B树、B+树的精确定义和区别探究

    概述 关于树的概念很多,B树,B+树,红黑树等等. 但是你去翻翻百度百科,或者用百度或者谷歌搜索一下中文的树结构的介绍,全都是狗屁.没有哪个中文网站是真正精确解释树的定义的,尤其是百度百科. 下面我要 ...

  5. MFC里 显示设备上下文CClient dc(this) 和 CPaintDC dc(this)

    1 CPaintDC类(1)CPaintDC类是CDC类的一个派生类,该类一般用在响应WM_PAINT消息的函数OnPaint()中.(2)WM_PAINT消息是当窗口的某个区域需要重画时激发的窗口消 ...

  6. 洛谷 P2735 电网

    https://www.luogu.org/problemnew/show/P2735 定理什么的最讨厌了,匹克定理?不会,也不想学. 粉色的为电网,将图中的电网我们将他构造一个矩形,然后蓝色和绿色的 ...

  7. BZOJ 2002 弹飞绵羊(分块)

    题目:弹飞绵羊 这道题,据说是lct裸题,但是lct那么高级的数据结构,我并不会,所以采取了学长讲过的分块做法,我们对序列分块,可以定义两个数组,其中一个表示从当前位置跳出当前块需要多少步,另一个数组 ...

  8. 201621123080 《Java程序设计》 第7周学习总结

    1. 本周学习总结 1.1 思维导图:Java图形界面总结 2.书面作业 1. GUI中的事件处理 1.1 写出事件处理模型中最重要的几个关键词. 事件 事件源 事件监听器 事件处理方法 1.2 任意 ...

  9. Beyond Compare4.x 破解方案

    如果过了30天的评估期或报“Beyond Compare 许可证密钥被撤销” 而导致不能再打开Beyond Compare,可以用下面的方法来解决: 1.找到“C:\Users\[Your User ...

  10. Thinkphp5 同时连接两个库

    新建api/user.php <?php /** * Created by PhpStorm. * User: Administrator * Date: 2018/8/25 * Time: 1 ...