[BZOJ5361]/[HDU6291]对称数
2024-08-27 03:35:41
[BZOJ5361]/[HDU6291]对称数
题目大意:
一个\(n(n\le2\times10^5)\)个结点的树,每个结点有一个权值\(a_i(a_i\le2\times10^5)\),\(m(m\le2\times10^5)\)次询问,每次询问\((u,v)\)路径上最小的出现偶数次的数。
思路:
对每个权值随机一个unsigned long long
作为一个新的权值,树上主席树维护区间异或和。
询问时在主席树上二分即可。
源代码:
#include<cstdio>
#include<cctype>
#include<vector>
#include<cstring>
#include<cstdlib>
#include<algorithm>
inline int getint() {
register char ch;
while(!isdigit(ch=getchar()));
register int x=ch^'0';
while(isdigit(ch=getchar())) x=(((x<<2)+x)<<1)+(ch^'0');
return x;
}
typedef unsigned long long uint64;
const int N=2e5+1,logN=40;
int a[N],v[N],anc[N][logN],dep[N],mex;
uint64 w[N],hsum[N];
std::vector<int> e[N];
inline void add_edge(const int &u,const int &v) {
e[u].push_back(v);
e[v].push_back(u);
}
inline int lg2(const float &x) {
return ((unsigned&)x>>23&255)-127;
}
class FotileTree {
#define mid ((b+e)>>1)
private:
struct Node {
uint64 val;
int left,right;
};
Node node[N*logN];
int sz,new_node(const int &p) {
node[++sz]=node[p];
return sz;
}
public:
void reset() {
sz=0;
}
int root[N];
void insert(int &p,const int &b,const int &e,const int &x) {
p=new_node(p);
node[p].val^=w[v[x]];
if(b==e) return;
if(x<=mid) insert(node[p].left,b,mid,x);
if(x>mid) insert(node[p].right,mid+1,e,x);
}
int query(const int &p,const int &q,const int &r,const int &s,const int &b,const int &e) const {
if((node[p].val^node[q].val^node[r].val^node[s].val)==(hsum[e]^hsum[b-1])) return mex;
if(b==e) return v[b];
const int tmp=query(node[p].left,node[q].left,node[r].left,node[s].left,b,mid);
if(tmp==mex) return query(node[p].right,node[q].right,node[r].right,node[s].right,mid+1,e);
return tmp;
}
#undef mid
};
FotileTree t;
void dfs(const int &x,const int &par) {
anc[x][0]=par;
dep[x]=dep[par]+1;
for(register int i=1;i<=lg2(dep[x]);i++) {
anc[x][i]=anc[anc[x][i-1]][i-1];
}
const int p=std::lower_bound(&v[1],&v[v[0]]+1,a[x])-v;
t.insert(t.root[x]=t.root[par],1,v[0],p);
for(unsigned i=0;i<e[x].size();i++) {
const int &y=e[x][i];
if(y==par) continue;
dfs(y,x);
}
}
inline int get_lca(int x,int y) {
if(dep[x]<dep[y]) std::swap(x,y);
for(register int i=lg2(dep[x]-dep[y]);i>=0;i--) {
if(dep[anc[x][i]]>=dep[y]) x=anc[x][i];
}
if(x==y) return x;
for(register int i=lg2(dep[x]);i>=0;i--) {
if(anc[x][i]!=anc[y][i]) {
x=anc[x][i];
y=anc[y][i];
}
}
return anc[x][0];
}
inline int query(const int &x,const int &y) {
const int lca=get_lca(x,y);
return std::min(t.query(t.root[x],t.root[y],t.root[lca],t.root[anc[lca][0]],1,v[0]),mex);
}
int main() {
srand(19961993);
for(register int T=getint();T;T--) {
memset(anc,0,sizeof anc);
const int n=getint(),m=getint();
for(register int i=1;i<=n;i++) {
v[i]=a[i]=getint();
while(w[a[i]]==0) {
w[a[i]]=(uint64)rand()*rand()*rand();
}
}
std::sort(&v[1],&v[n]+1);
v[0]=std::unique(&v[1],&v[n]+1)-v-1;
for(register int i=mex=1;i<=v[0];i++) {
hsum[i]=hsum[i-1]^w[v[i]];
if(v[i]==mex) mex++;
}
for(register int i=1;i<n;i++) {
add_edge(getint(),getint());
}
dfs(1,0);
for(register int i=0;i<m;i++) {
printf("%d\n",query(getint(),getint()));
}
t.reset();
for(register int i=1;i<=n;i++) {
e[i].clear();
}
}
return 0;
}
最新文章
- springmvc注解事例
- 一种基于annotation的Spring-mvc权限控制方法
- Eclipse RCP实用小技巧
- 如何在UIimageview里显示一张图片里的某一部分
- codeforces CF475 ABC 题解
- 【转】在C#用HttpWebRequest中发送GET/HTTP/HTTPS请求
- Css杂谈
- Android百度地图开发03之地图控制 + 定位
- 大数据应用之:MongoDB从入门到精通你不得不知的21个为什么?
- HTML与CSS入门——第八章 使用外部和内部链接
- get方式提交中文乱码(两次编码,一次解码)
- 一丶Http协议
- CM记录-操作系统调优
- java poi处理excel多sheet并实现排序
- 转: Xshell鼠标选中,终端立即中断(CTRL-C)的问题
- ASP代码审计学习笔记 -5.文件下载漏洞
- eclipse打包jar文件
- 转---秒杀多线程第八篇 经典线程同步 信号量Semaphore
- 2.1 一个简单的Web工程例子
- Ubuntu电源键软关机设置
热门文章
- leetcode.C.4. Median of Two Sorted Arrays
- i春秋第二届春秋欢乐赛RSA256writeup
- $NTT$(快速数论变换)
- python的sorted函数对字典按value进行排序
- [How to] HBase的bulkload使用方法
- zookeeper客户端连接报错
- rpmdb: Thread/process 9180/139855524558592 failed: Thread died in Berkeley DB library
- MySQL 视图、触发器、函数、存储过程
- NTP路由器配置
- ceph rgw java sdk 使用域名访问服务时需要设置s3client的配置项 PathStyleAccess 为true, 负责将报域名异常