洛谷 P4592 [TJOI2018]异或 解题报告
P4592 [TJOI2018]异或
题目描述
现在有一颗以\(1\)为根节点的由\(n\)个节点组成的树,树上每个节点上都有一个权值\(v_i\)。现在有\(Q\)次操作,操作如下:
1 x y
:查询节点\(x\)的子树中与\(y\)异或结果的最大值2 x y
:查询路径\(x\)到\(y\)上点与\(z\)异或结果最大值
输入输出格式
输入格式:
第一行是两个数字\(n,Q\);
第二行是\(n\)个数字用空格隔开,第\(i\)个数字\(v_i\)表示点\(i\)上的权值
接下来\(n-1\)行,每行两个数,\(x,y\),表示节点\(x\)与\(y\)之间有边
接下来\(Q\)行,每一行为一个查询,格式如上所述.
输出格式:
对于每一个查询,输出一行,表示满足条件的最大值。
说明
对于\(10\%\)的数据,有\(1<n,Q\leq100\)
对于\(20\%\)的数据,有\(1<n,Q\leq1000\)
对于\(40\%\)的数据,有\(1<n,Q\leq10000\)
对于\(100\%\)的数据,有\(1<n,Q\leq100000\)
对于\(100\%\)的数据,有查询\(1\)中的\(y\leq2^{30}\),查询\(2\)中的\(z\leq2^{30}\)。
区间异或最大值可以用可持久化字典树实现。
这个题建对DFS序建一颗,对自根向下的链建,分别处理两种询问就可以了。
Code:
#include <cstdio>
#include <cctype>
const int N=1e5+10;
int read()
{
int x=0;char c=getchar();
while(!isdigit(c)) c=getchar();
while(isdigit(c)) {x=x*10+c-'0';c=getchar();}
return x;
}
int max(int x,int y){return x>y?x:y;}
#define ls ch[now][0]
#define rs ch[now][1]
#define rep(i,a,b) for(int i=a;i<=b;i++)
int head[N],to[N<<1],Next[N<<1],cnt;
void add(int u,int v)
{
to[++cnt]=v,Next[cnt]=head[u],head[u]=cnt;
}
int poi[N],n,Q;
namespace work1//子树
{
int ch[N*32][2],mx[N*32],tot,dfn[N],low[N],ha[N],dfsclock,root[N];
void dfs(int now,int fa)
{
dfn[now]=++dfsclock;
ha[dfsclock]=now;
for(int i=head[now];i;i=Next[i])
if(to[i]!=fa)
dfs(to[i],now);
low[now]=dfsclock;
}
void Insert(int las,int &now,int dep,int id)
{
if(!now) now=++tot;
if(dep<0){mx[now]=id;return;}
int bit=poi[ha[id]]>>dep&1;
Insert(ch[las][bit],ch[now][bit],dep-1,id);
ch[now][bit^1]=ch[las][bit^1];
mx[now]=max(mx[ls],mx[rs]);
}
void init()
{
dfs(1,0);
rep(i,1,n) Insert(root[i-1],root[i],30,i);
}
int query(int now,int les,int dep,int x)
{
if(dep<0) return x^poi[ha[mx[now]]];
int bit=x>>dep&1;
if(mx[ch[now][bit^1]]>=les) return query(ch[now][bit^1],les,dep-1,x);
return query(ch[now][bit],les,dep-1,x);
}
void work()
{
int x=read(),y=read();
printf("%d\n",query(root[low[x]],dfn[x],30,y));
}
}
namespace work2
{
int f[N][20],ch[N*32][2],mx[N*32],root[N],dep[N],en[N*32],tot,tmp;
void swap(int &x,int &y){tmp=x,x=y,y=tmp;}
void Insert(int las,int &now,int de,int id)
{
if(!now) now=++tot;
if(de<0){mx[now]=dep[id],en[now]=id;return;}
int bit=poi[id]>>de&1;
Insert(ch[las][bit],ch[now][bit],de-1,id);
ch[now][bit^1]=ch[las][bit^1];
mx[now]=max(mx[ls],mx[rs]);
}
void dfs(int now)
{
for(int i=1;f[now][i-1];i++) f[now][i]=f[f[now][i-1]][i-1];
for(int i=head[now];i;i=Next[i])
{
int v=to[i];
if(v==f[now][0]) continue;
dep[v]=dep[now]+1;
f[v][0]=now;
Insert(root[now],root[v],30,v);
dfs(v);
}
}
void init()
{
dep[1]=1;
Insert(0,root[1],30,1);
dfs(1);
}
int LCA(int x,int y)
{
if(dep[x]<dep[y]) swap(x,y);
for(int i=18;~i;i--)
if(dep[f[x][i]]>=dep[y])
x=f[x][i];
if(x==y) return x;
for(int i=18;~i;i--)
if(f[x][i]!=f[y][i])
x=f[x][i],y=f[y][i];
return f[x][0];
}
int query(int now,int les,int de,int x)
{
if(de<0) return poi[en[now]]^x;
int bit=x>>de&1;
if(mx[ch[now][bit^1]]>=les) return query(ch[now][bit^1],les,de-1,x);
return query(ch[now][bit],les,de-1,x);
}
void work()
{
int x=read(),y=read(),z=read(),lca=LCA(x,y);
printf("%d\n",max(query(root[x],dep[lca],30,z),query(root[y],dep[lca],30,z)));
}
}
int main()
{
n=read(),Q=read();
rep(i,1,n) scanf("%d",poi+i);
for(int u,v,i=1;i<n;i++)
u=read(),v=read(),add(u,v),add(v,u);
work1::init();
work2::init();
rep(i,1,Q)
{
if(read()==1) work1::work();
else work2::work();
}
return 0;
}
2018.11.3
洛谷 P4592 [TJOI2018]异或 解题报告的更多相关文章
- [洛谷P4592][TJOI2018]异或
题目大意:有一棵$n$个点的树,第$i$个点权值为$w_i$,有两种操作: $1\;x\;y:$询问节点$x$的子树中与$y$异或结果的最大值 $2\;x\;y\;z:$询问路径$x$到$y$上点与$ ...
- 洛谷P4592 [TJOI2018]异或(可持久化01Trie)
题意 题目链接 可持久化01Trie板子题 对于两个操作分别开就行了 #include<bits/stdc++.h> using namespace std; const int MAXN ...
- 洛谷P4592 [TJOI2018]异或 【可持久化trie树】
题目链接 BZOJ4592 题解 可持久化trie树裸题 写完就A了 #include<algorithm> #include<iostream> #include<cs ...
- 洛谷_Cx的故事_解题报告_第四题70
1.并查集求最小生成树 Code: #include <stdio.h> #include <stdlib.h> struct node { long x,y,c; ...
- 洛谷 P2317 [HNOI2005]星际贸易 解题报告
P2317 [HNOI2005]星际贸易 题目描述 输入输出格式 输入格式: 输出格式: 如果可以找到这样的方案,那么输出文件output.txt中包含两个整数X和Y.X表示贸易额,Y表示净利润并且两 ...
- 洛谷 P3802 小魔女帕琪 解题报告
P3802 小魔女帕琪 题目背景 从前有一个聪明的小魔女帕琪,兴趣是狩猎吸血鬼. 帕琪能熟练使用七种属性(金.木.水.火.土.日.月)的魔法,除了能使用这么多种属性魔法外,她还能将两种以上属性组合,从 ...
- 洛谷 P2606 [ZJOI2010]排列计数 解题报告
P2606 [ZJOI2010]排列计数 题目描述 称一个\(1,2,...,N\)的排列\(P_1,P_2...,P_n\)是\(Magic\)的,当且仅当对所以的\(2<=i<=N\) ...
- 洛谷1303 A*B Problem 解题报告
洛谷1303 A*B Problem 本题地址:http://www.luogu.org/problem/show?pid=1303 题目描述 求两数的积. 输入输出格式 输入格式: 两个数 输出格式 ...
- 洛谷 P4074 [WC2013]糖果公园 解题报告
P4074 [WC2013]糖果公园 糖果公园 树上待修莫队 注意一个思想,dfn序处理链的方法,必须可以根据类似异或的东西,然后根据lca分两种情况讨论 注意细节 Code: #include &l ...
随机推荐
- 理解Python的装饰器
看Flask文档时候看到关于cache的装饰器,有这么一段代码: def cached(timeout=5 * 60, key=’view/%s’): def decorator(f): @wraps ...
- java 前后端 日期转换
1.前传后 @DateTimeFormat(pattern="yyyy-MM-dd") private Date payTime; 2.后传前 @JsonFormat(patter ...
- Qt-QML-Charts-ChartView-编译错误-ASSERT: "!"No style available without QApplication!
昨天本来是回家想好好琢磨一下使用Chart来绘制曲线的,奈何在建立项目的时候也就卡住了,加上心情比较烦躁,也没有耐心寻找答案就草草了事.所以今天继续搞定这个. 上图是Qt 的编译错误截图 QML de ...
- div布局方案整理
实际项目开发过程中遇到页面 DIV 左右布局的需求:左侧 DIV 固定宽度,右侧 DIV 自适应宽度,填充满剩余页面,由此引申出本文的几种解决方案 1 左侧 DIV 设置 float 属性为 left ...
- 【token接口】-jmeter
token 接口 3步骤 1.登录接口 2.提取登录接口的token 3.http 信息管理头 把提取的cookie传入 就可以了
- 【转】MMORPG游戏服务器技能系统设计:表格字段与技能程序框架
本文主要从一个程序员的角度阐述一下mmorpg服务器技能系统的程序框架设计,最近在做这个,就当做一个总结吧,其中某些概念可能没有解释清楚,欢迎大家拍砖讨论~ 技能其实是战斗系统的一个组成部分,战斗基本 ...
- Git版本库工作流程图想
对照廖雪峰的教程,发现有很多难以理解的地方,画了一个图想方便以后参考 首先两个基本命令反应了版本库最本质的工作流程,后面的命令其实都基于此git add 把文件修改添加到暂存区git commit 在 ...
- struts2之form标签theme属性详解
struts2中theme属性包括xhtml,html,simple,ajax .默认是xhtml theme:设置struts2标签的主题,默认为xhtml. theme=xhtml时:会默认额外生 ...
- js学习之正则表达式
js学习之正则表达式 正则表达式(英语:Regular Expression,在代码中常简写为regex.regexp或RE)使用单个字符串来描述.匹配一系列符合某个句法规则的字符串搜索模式 一:语法 ...
- svn服务器 备份,迁移,部署方案
这次做业务迁移,要从一个云厂商迁移到某云厂商,之前每天到全备svn排到用场了,需要搭建一个全新到svn服务并要做迁移,并实现我们开发机到时时代码同步 一.svn备份有很多种,优劣都不同,百度可查,我采 ...