Problem A: 种树

Description

很久很久以前,一个蒟蒻种了一棵会提问的树,树有\(n\)个节点,每个节点有一个权值,现在树给出\(m\)组询问,每次询问两个值:树上一组点对\((x,y)\)简单路径上不同权值的数量以及权值的\(\max\)为多少

然而某一天毒瘤胖子过来给树浇了点水,询问变成了每次求,一组点对\((x,y)\)简单路径上的不同权值的数量以及权值的\(mex\)

然而又过了两天毒瘤袁稳稳也过来给树浇了点水,询问变成了每次求若干组点对\((x,y)\)简单路径的并集上的不同权值的数量以及权值的\(mex\)(如有疑惑见\(HINT\))

然后蒟蒻就菜哭在树下了qwq并且毫不负责任地把这个问题丢给了刚好路过的你

因为这棵树受到了两个毒瘤的祝福,每次询问受到了加密,记\(lastans\)表示上一次询问的两个答案的和,这次询问中的读入的表示点对的两个数都要\(xor \ (lastans∗op)\),其中\(op\in \{0,1\}\),具体见数据范围

后话:然而毕竟是蒟蒻种的树,毒瘤的祝福并没有使这题送温暖的本质发生变化qwq

Input

第一行三个整数\(n,m,op\)

接下来一行\(n\)个整数表示每个节点的权值\(val_i\)

再接下来\(n−1\)行每行两个整数\(x,y\)表示树上的一条边

再接下来\(m\)组询问,每组询问第一行一个整数\(num\)表示点对的数量,接下来\(num\)行每行两个整数\((x,y)\)表示一组点对

Output

对于每组询问,输出一行两个整数分别表示不同权值的数量以及权值的\(mex\)

Sample Input

5 5 0
2 0 0 1 3
1 2
2 3
2 4
4 5
1 4 5
3 1 5 5 2 4 4
2 2 4 2 4
4 2 5 3 1 4 3 2 5
1 2 5

Sample Output

2 0
4 4
2 2
4 4
3 2

HINT

一些你可能根本不需要用到的说明:一个数集\(S\)的\(mex\)为最小的满足\(x\notin S\)的非负整数\(x\)

\(subtask1(20\%)\):\(n,m≤1000,\sum num≤1000,op=0\)

\(subtask2(30\%)\):\(n,m≤10^5,\sum num≤10^5\),树是一条链,\(op=0\)

\(subtask3(50\%)\):\(n,m≤10^5,\sum num≤10^5,0≤val_i≤30000\)


完 全 没 有 感 受 到 温 暖,虽 然 确 实 是 最 简 单 的 一 道,剩 下 两 道 我 改 不 出 来

30000这个数我们很容易除上个64哎

然后随便用树剖倍增之类的维护一下,发现单次操作复杂度达到了惊人的\(\log n\frac{val}{64}\),显然没救了。

这时候就是分块出场的时候辣

因为询问的是链的信息,所以我们考虑对树的深度进行分块,既对树提取一定的关键点,相邻的关键点深度差不超过\(\sqrt n\)就可以了,这样我们就有了\(\sqrt n\)个关键点。

然后我们拿关键点拼吗?复杂度达到了更惊人的\(\sqrt n \frac{val}{64}\)

所以考虑先预处理在同一条到跟路径上关键点的路径信息,这里采用手写\(bitset\)的方法就可以做到\(O(1)\)整数与上\(bitset\)了,然后每个关键点向上与顺便更新一下就行了。

查询的时候,不完整的暴力跳,完整的做一次bitset之间的与运算就行了。

复杂度:\(O(n\sqrt n+q(\sqrt n+\frac{val}{64}))\)


Code:

#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
#define ull unsigned long long
const int N=1e5+10,B=470;
const ull full=~0ull,cut=(1ull<<16)-1;
int ct[cut+1],n,m,op;
int cal(ull x){return ct[x&cut]+ct[x>>16&cut]+ct[x>>32&cut]+ct[x>>48&cut];}
using std::max;
struct Bitset
{
ull dx[B];
int len;
void clear(){memset(dx,0,sizeof(dx)),len=0;}
Bitset(){clear();}
void friend operator |=(Bitset &A,int x){A.dx[x>>6]|=1ull<<(x&63);A.len=max(A.len,x>>6);}
void friend operator |=(Bitset &A,Bitset B)
{
A.len=A.len>B.len?A.len:B.len;
for(int i=0;i<=A.len;i++) A.dx[i]|=B.dx[i];
}
int count()
{
int ret=0;
for(int i=0;i<=len;i++)
ret+=cal(dx[i]);
return ret;
}
int mex()
{
for(int i=0;i<=len;i++)
{
if(dx[i]==full) continue;
for(int j=0;j<64;j++)
if(!(dx[i]>>j&1))
return (i<<6)+j;
}
return 233;
}
}path[320][320],ans;
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 f[N][19],dep[N],mxdep[N],id[N],rt[N],val[N],pre[N],H;
void dfs(int now)
{
for(int i=1;f[now][i-1];i++) f[now][i]=f[f[now][i-1]][i-1];
dep[now]=dep[f[now][0]]+1;
mxdep[now]=1;
for(int v,i=head[now];i;i=Next[i])
if((v=to[i])!=f[now][0])
{
f[v][0]=now;
dfs(v);
mxdep[now]=max(mxdep[now],mxdep[v]+1);
}
if(mxdep[now]==H||now==1)
{
rt[id[now]=++rt[0]]=now;
mxdep[now]=0;
}
}
int LCA(int x,int y)
{
if(dep[x]<dep[y]) return LCA(y,x);
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];
}
void query(int x,int y)
{
while(!id[x]&&x!=y) ans|=val[x],x=f[x][0];
if(x==y) {ans|=val[x];return;}
int s=x;
while(dep[pre[x]]>dep[y]) x=pre[x];
ans|=path[id[s]][id[x]];
while(dep[f[x][0]]>=dep[y]) x=f[x][0],ans|=val[x];
}
void Query(int x,int y)
{
int lca=LCA(x,y);
query(x,lca);
query(y,lca);
}
int main()
{
scanf("%d%d%d",&n,&m,&op);
for(int i=1;i<=cut;i++) ct[i]=ct[i>>1]+(i&1);
H=sqrt(n)+1;
for(int i=1;i<=n;i++) scanf("%d",val+i);
for(int u,v,i=1;i<n;i++) scanf("%d%d",&u,&v),add(u,v),add(v,u);
dfs(1);
for(int i=1;i<=rt[0];i++)
{
Bitset tmp;
tmp|=val[rt[i]];
path[i][i]=tmp;
for(int now=f[rt[i]][0];now;now=f[now][0])
{
tmp|=val[now];
if(id[now])
{
path[i][id[now]]=tmp;
if(!pre[rt[i]]) pre[rt[i]]=now;
}
}
}
for(int lastans=0,num,x,y,i=1;i<=m;i++)
{
scanf("%d",&num);
ans.clear();
for(int j=1;j<=num;j++)
{
scanf("%d%d",&x,&y);
x^=lastans*op,y^=lastans*op;
Query(x,y);
}
int t1=ans.count(),t2=ans.mex();
lastans=t1+t2;
printf("%d %d\n",t1,t2);
}
return 0;
}

2019.1.6

Problem A: 种树 解题报告的更多相关文章

  1. ZOJ Problem Set - 1025解题报告

    ZOJ Problem Set - 1025 题目分类:基础题 原题地址:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=10 ...

  2. ACM: A Simple Problem with Integers 解题报告-线段树

    A Simple Problem with Integers Time Limit:5000MS Memory Limit:131072KB 64bit IO Format:%lld & %l ...

  3. BestCoder18 1002.Math Problem(hdu 5105) 解题报告

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5105 题目意思:给出一个6个实数:a, b, c, d, l, r.通过在[l, r]中取数 x,使得 ...

  4. Problem - 433C - Codeforces解题报告

    对于这题本人刚开始的时候的想法是:先把最大两数差的位置找到然后merge计算一个值再与一连串相同的数做merge后计算一个值比较取最大值输出:可提交后发现不对,于是本人就搜了一下正解发现原来这题的正确 ...

  5. Problem A: 选举 解题报告

    Problem A: 选举 题意 给出一个投票过程.有\(n\)个选民和\(m\)个候选人,每个选民\(i\)有个不重且有序的可投集合\(\{a_i\}\). 对于第一轮投票,选民\(i\)会投给\( ...

  6. Problem C: 多线程 解题报告

    Problem C: 多线程 Description 多线程是一种常见的加速手段,利用多个线程同时处理不同的任务可以一定程度上减少总耗时,达到提高效率的目的.然而,多个线程间的执行顺序是完全不可控的, ...

  7. Problem A: 踢罐子 解题报告

    Problem A: 踢罐子 Description 平面上有\(n\)个点,其中任意2点不重合,任意3点不共线. 我们等概率地选取一个点A,再在剩下的\(n-1\)个点中等概率地选取一个点B,再在剩 ...

  8. Problem B: 专家系统 解题报告

    Problem B: 专家系统 Description 一个专家系统是指,你雇佣了\(n\)个专家,他们每个人会做出一个结果,然后你从中选取较多的专家的结果组合而成最终的结果.专家系统广泛应用于传统机 ...

  9. Problem C Dist 解题报告

    Problem C Dist Description 有一个\(n\)个点带边权的连通无向图,边集用\(k\)个集合\(s_1,s_2,\dots,s_k\)和\(k\)个整数\(w_1,w_2,\d ...

随机推荐

  1. 判断库位是否参与MRP运算

    表 T001L 字段DISKZ (库存地点MRP标识)为空,参与MRP运算,为1不参与.

  2. 2017-2018 Exp9 网络欺诈技术防范 20155214

    目录 Exp9 网络欺诈技术防范 实验内容 Webgoat General Access Control Flaws Crossing-Site Scripting Injection Flaws 知 ...

  3. spring配置多个事务管理器

    <tx:annotation-driven/> <bean id="transactionManager1" class="org.springfram ...

  4. LOJ#6354. 「CodePlus 2018 4 月赛」最短路[最短路优化建图]

    题意 一个 \(n\) 个点的完全图,两点之间的边权为 \((i\ xor\ j)*C\) ,同时有 \(m\) 条额外单向路径,问从 \(S\) 到 \(T\) 的最短路. \(n\leq 10^5 ...

  5. binary 和 varbinary 用法全解

    在SQL Server中,使用数据类型 binary(n) 和 varbinary(n) 存储二进制数据,n是指字节数量: binary(n):固定长度为 n 字节,其中 n 值从 1 到 8,000 ...

  6. flask_admin 笔记二 授权和权限

    权限当然就是让有应该权限的用户能执行某些操作,把没有权限的用户限制在外面.Flask-admin提供了几种方法来处理: 1, Http basic Auth 最简单的身份验证形式是HTTP基本身份验证 ...

  7. LintCode——交叉字符串

    描述:给出三个字符串:s1.s2.s3,判断s3是否由s1和s2交叉构成. 样例:s1 = "aabcc" s2 = "dbbca" - 当 s3 = &quo ...

  8. bower配置私服nexus3

    内容来自 https://help.sonatype.com/repomanager3/bower-repositories#BowerRepositories-BrowsingBowerReposi ...

  9. VC++ MFC程序设置以管理员权限运行

    1.引入资源文件 (.manifest文件),文件中的 name 值为程序全称.exe:文件内容如下: <?xml version="1.0" encoding=" ...

  10. PHP 观察者模式和php实现 Observer Pattern

    观察者模式: 观察者模式(Observer Pattern):定义对象间的一种一对多依赖关系,使得每当一个对象状态发生改变时,其相关依赖对象皆得到通知并被自动更新.观察者模式又叫做发布-订阅(Publ ...