Problem A: 种树 解题报告
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: 种树 解题报告的更多相关文章
- ZOJ Problem Set - 1025解题报告
ZOJ Problem Set - 1025 题目分类:基础题 原题地址:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=10 ...
- ACM: A Simple Problem with Integers 解题报告-线段树
A Simple Problem with Integers Time Limit:5000MS Memory Limit:131072KB 64bit IO Format:%lld & %l ...
- 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,使得 ...
- Problem - 433C - Codeforces解题报告
对于这题本人刚开始的时候的想法是:先把最大两数差的位置找到然后merge计算一个值再与一连串相同的数做merge后计算一个值比较取最大值输出:可提交后发现不对,于是本人就搜了一下正解发现原来这题的正确 ...
- Problem A: 选举 解题报告
Problem A: 选举 题意 给出一个投票过程.有\(n\)个选民和\(m\)个候选人,每个选民\(i\)有个不重且有序的可投集合\(\{a_i\}\). 对于第一轮投票,选民\(i\)会投给\( ...
- Problem C: 多线程 解题报告
Problem C: 多线程 Description 多线程是一种常见的加速手段,利用多个线程同时处理不同的任务可以一定程度上减少总耗时,达到提高效率的目的.然而,多个线程间的执行顺序是完全不可控的, ...
- Problem A: 踢罐子 解题报告
Problem A: 踢罐子 Description 平面上有\(n\)个点,其中任意2点不重合,任意3点不共线. 我们等概率地选取一个点A,再在剩下的\(n-1\)个点中等概率地选取一个点B,再在剩 ...
- Problem B: 专家系统 解题报告
Problem B: 专家系统 Description 一个专家系统是指,你雇佣了\(n\)个专家,他们每个人会做出一个结果,然后你从中选取较多的专家的结果组合而成最终的结果.专家系统广泛应用于传统机 ...
- Problem C Dist 解题报告
Problem C Dist Description 有一个\(n\)个点带边权的连通无向图,边集用\(k\)个集合\(s_1,s_2,\dots,s_k\)和\(k\)个整数\(w_1,w_2,\d ...
随机推荐
- Php的常见错误及错误分析
我们在进行开发工作的时候,难免会遇到PHP的报错,解决这些错误,也是作为PHPer必须掌握的一种技能. 如果程序发生错误,我们能大致的分析出出现错误的原因,对于我们解决这戏错误会有很大的帮助. Not ...
- UWP-开发一个具有闹钟,天气预报,翻译,语音功能的Demo
UWP即Windows 10中的Universal Windows Platform简称.即Windows通用应用平台,在Windows 10 Mobile/Surface(Windows平板电脑)/ ...
- mfc CImageList和CListCtrl
知识点: CImageList类的运用 CListCtrl添加图标 一.CImageList CImageList*SetImageList(CImageList*pImageList,int nIm ...
- Photoshop CS4破解方法
先在网上下载Photoshop CS4的版本,安装后按如下步骤操作即可. 激活码: 1330-1082-3503-2270-3738-6738 1330-1776-8671-6289-7706-291 ...
- selenium常用命令
openopen(url)- 在浏览器中打开URL,可以接受相对和绝对路径两种形式type type(inputLocator, value)- 模拟人手的输入过程,往指定的input中输入值- 也适 ...
- effective c++ 笔记 (30-31)
//---------------------------15/04/17---------------------------- //#30 透彻了解inlineing的里里外外 { /* 1: ...
- NodeJS旅程 : module 不可忽略的重点
modules 模块的简介 Module 是Node.js中最重要的一个部分也是进行深度开发前的必修课.掌握Module才能真正理解NodeJS的精髓,你会发现从思路上会有极大的扩展. 学会写mod ...
- 我用Python爬虫挣钱的那些事
在下写了10年Python,期间写了各种奇葩爬虫,挣各种奇葩的钱,写这篇文章总结下几种爬虫挣钱的方式. 1.最典型的就是找爬虫外包活儿. 这个真是体力活,最早是在国外各个freelancer网站上找适 ...
- Ubuntu命令行运行C程序和C++程序
首先Ctrl + T 打开一个终端,cd到你建立C/C++文件的目录下. 下面以建立 helloc.c 和 hellocpp.cpp 进行演示 vim helloc.c 按 i 进入插入操作,然后写C ...
- Cloud Native Weekly | Kubernetes 1.13发布
云原生一周精选 1——Kubernetes 1.13发布 2——Kubernetes首次出现重大安全漏洞 3——Docker和微软公司推出云原生应用的部署规范 4——谷歌推出beta版本的Cloud ...