[BZOJ5361][Lydsy1805月赛]对称数
Description
给你一棵树,每个点有一个编号\(a_i\)。\(Q\)组询问,每次问一条路径上最小的出现了偶数次的编号是多少(包括零次)。
多组数据,\(T\le10,n,Q,a_i\le200000\)
sol
这又是一道随机化神题。
给每个编号随机一个\(unsigned\ long\ long\)范围内的权值\(val_i\)。
对于一次询问,统计路径上所有编号\(\in [l,mid]\)的点的权值的异或和。如果这个异或和不等于\(val_l\otimes val_{l+1}\otimes...\otimes val_{mid}\),就说明这个\([l,mid]\)内至少有一个编号出现了偶数次,否则说明所有编号都出现了奇数次。建立主席树后在主席树上二分即可。
code
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<ctime>
using namespace std;
int gi(){
int x=0,w=1;char ch=getchar();
while ((ch<'0'||ch>'9')&&ch!='-') ch=getchar();
if (ch=='-') w=0,ch=getchar();
while (ch>='0'&&ch<='9') x=(x<<3)+(x<<1)+ch-'0',ch=getchar();
return w?x:-x;
}
#define ull unsigned long long
const int N = 2e5+5;
struct president_tree{int ls,rs;ull sum;}t[N*25];
int n,m,to[N<<1],nxt[N<<1],head[N],cnt,col[N],mx;
int fa[N],dep[N],sz[N],son[N],top[N],rt[N],tot;
ull val[N],sum[N];
void link(int u,int v){
to[++cnt]=v;nxt[cnt]=head[u];head[u]=cnt;
}
void modify(int &x,int l,int r,int p){
t[++tot]=t[x];t[x=tot].sum^=val[p];
if (l==r) return;int mid=l+r>>1;
if (p<=mid) modify(t[x].ls,l,mid,p);
else modify(t[x].rs,mid+1,r,p);
}
int query(int x,int y,int z,int w,int l,int r){
if (l==r) return l;int mid=l+r>>1;
if ((t[t[x].ls].sum^t[t[y].ls].sum^t[t[z].ls].sum^t[t[w].ls].sum)!=(sum[mid]^sum[l-1]))
return query(t[x].ls,t[y].ls,t[z].ls,t[w].ls,l,mid);
else return query(t[x].rs,t[y].rs,t[z].rs,t[w].rs,mid+1,r);
}
void dfs1(int u,int f){
fa[u]=f;dep[u]=dep[f]+1;sz[u]=1;
modify(rt[u]=rt[f],1,mx,col[u]);
for (int e=head[u];e;e=nxt[e])
if (to[e]!=f){
dfs1(to[e],u),sz[u]+=sz[to[e]];
if (sz[to[e]]>sz[son[u]]) son[u]=to[e];
}
}
void dfs2(int u,int up){
top[u]=up;if (son[u]) dfs2(son[u],up);
for (int e=head[u];e;e=nxt[e])
if (to[e]!=fa[u]&&to[e]!=son[u])
dfs2(to[e],to[e]);
}
int lca(int u,int v){
while (top[u]^top[v]){
if (dep[top[u]]<dep[top[v]]) swap(u,v);
u=fa[top[u]];
}
return dep[u]<dep[v]?u:v;
}
int main(){
srand(20020415);
for (int i=1;i<=200001;++i) val[i]=(ull)rand()*rand()*rand(),sum[i]=sum[i-1]^val[i];
int T=gi();while (T--){
n=gi();m=gi();mx=0;
memset(head,0,sizeof(head));cnt=0;
memset(t,0,sizeof(t));tot=0;
memset(son,0,sizeof(son));
memset(rt,0,sizeof(rt));
for (int i=1;i<=n;++i) col[i]=gi(),mx=max(mx,col[i]);++mx;
for (int i=1;i<n;++i){
int u=gi(),v=gi();
link(u,v),link(v,u);
}
dfs1(1,0),dfs2(1,1);
while (m--){
int u=gi(),v=gi(),gg=lca(u,v);
printf("%d\n",query(rt[u],rt[v],rt[gg],rt[fa[gg]],1,mx));
}
}
return 0;
}
[BZOJ5361][Lydsy1805月赛]对称数的更多相关文章
- 【主席树上二分】bzoj5361: [Lydsy1805月赛]对称数
随机化选讲例题 题目大意 小 Q 认为,偶数具有对称美,而奇数则没有.给定一棵 n 个点的树,任意两点之间有且仅有一条直接或间接路径.这些点编号依次为 1 到 n,其中编号为 i 的点上有一个正整数 ...
- BZOJ5361[Lydsy1805月赛]对称数——主席树+随机化
题目链接:https://www.lydsy.com/JudgeOnline/problem.php?id=5361 好神的一道题啊! 容易看出来是要用维护权值的数据结构,因此树链剖分首先pass掉. ...
- [Lydsy1805月赛]对称数 BZOJ5361
分析: 这个题,还是蛮有趣的.考虑,如果l,r区间内的所有数出现奇数次,那么[l-1,r]的抑或和等于所得抑或和. 之后怎么维护呢,主席树维护区间抑或和,记得将每个点附上一个ull级别的随机数,之后抑 ...
- [Lydsy1805月赛] 对称数
挺不错的一道数据结构题QWQ. 一开始发现这个题如果不看数据范围的话,妥妥的树上莫队啊23333,然鹅10组数据是不可能让你舒舒服服的树上莫队卡过的23333 于是想了想,这个题的模型就是,把u到v链 ...
- [BZOJ5361]/[HDU6291]对称数
[BZOJ5361]/[HDU6291]对称数 题目大意: 一个\(n(n\le2\times10^5)\)个结点的树,每个结点有一个权值\(a_i(a_i\le2\times10^5)\),\(m( ...
- [Bzoj5358][Lydsy1805月赛]口算训练(预处理+动态开点线段树)
5358: [Lydsy1805月赛]口算训练 Time Limit: 5 Sec Memory Limit: 512 MBSubmit: 318 Solved: 105[Submit][Stat ...
- [LeetCode] Strobogrammatic Number III 对称数之三
A strobogrammatic number is a number that looks the same when rotated 180 degrees (looked at upside ...
- [LeetCode] Strobogrammatic Number II 对称数之二
A strobogrammatic number is a number that looks the same when rotated 180 degrees (looked at upside ...
- [LeetCode] Strobogrammatic Number 对称数
A strobogrammatic number is a number that looks the same when rotated 180 degrees (looked at upside ...
随机推荐
- #if 0 #endif && #if 1 #endif ----待整理
在看一个 usbcan 的上位机例程中发现了这个,于是百度下,记录下来.(参考:http://nevel.cnblogs.com/p/6378035.html)
- 2016ACM/ICPC亚洲区沈阳站H - Guessing the Dice Roll HDU - 5955 ac自动机+概率dp+高斯消元
http://acm.hdu.edu.cn/showproblem.php?pid=5955 题意:给你长度为l的n组数,每个数1-6,每次扔色子,问你每个串第一次被匹配的概率是多少 题解:先建成ac ...
- python高级内置函数和各种推导式的介绍:一行搞定的代码
一.知识要点 all 都为真 any 有真的 min 最小的 max 最大的 sum 求和 reversed 反转 sorted 排序 zip 对应合并 [] 列表推倒式 () 生成器 {} 字典推倒 ...
- C++复习10.对象的初始化拷贝析构函数
对象的初始化.拷贝构造和析构函数 20131002 构造函数.析构函数.赋值函数是类的基本函数.每一个类只有一个析构函数,但是可以有多个构造函数.多个赋值函数.一般如果类中没有显示的声明和定义上述函数 ...
- hdu3718
题解: 见图 按照每一个位置上有相同加一 然后km 代码: #include<cstdio> #include<cmath> #include<cstring> # ...
- hdu 6097 Mindis(数学几何,圆心的反演点)
Mindis Time Limit: 6000/3000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others)Total Subm ...
- hdu 6038 Function
Function Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 131072/131072 K (Java/Others)Total ...
- 【html】META http-equiv 大全
meta是html语言head区的一个辅助性标签.几乎所有的网页里,我们可以看到类似下面这段的html代码: <head><meta http-equiv="content ...
- AOP的Advice
@Before 方法执行之前执行 @AfterReturning 方法正常执行完成后执行 @AfterThrowing 抛出任何异常之后执行 @After 就是相当于finally,它会将你的方法t ...
- 第9课:备份mysql数据库、重写父类、unittest框架、多线程
1. 写代码备份mysql数据库: 1)Linux下,备份mysql数据库,在shell下执行命令:mysqldump -uroot -p123456 -A >db_bak.sql即可 impo ...