题目链接:糖果公园

  听说这是一道树上莫队的入门题,于是我就去写了……顺便复习了一下莫队的各种姿势。

  首先,我们要在树上使用莫队,那么就需要像序列一样给树分块。这个分块的过程就是王室联邦这道题(vfleaking大神的博客里也有讲),当然也可以按照\(dfs\)序什么的进行分块。

  但我还是想在这里讲一下我的分块做法(第一种)。我们可以先设一个块的大小$S$,然后对整棵树进行$dfs$。每次我们$dfs$完一棵子树回溯时,我们就把这个点$u$加入到一个队列中,更新$fa_u$的$siz$。这里$siz_u$记录的是以$u$为根的子树中为分好块的节点数。当我们发现$siz_u$超过了$S$时,我们就可以给这些节点新建一个块。最后还会剩下不超过$S$个节点,我们可以新建一个块,也可以把这些点丢到最后新建的那个块中。如果采用后一种方式,那么就保证了每个块的大小都在$[S,3S)$之间且最多只有一个块的大小会大于$2S$。于是我们就得到了一种优美的分块方法。(如果还不懂的话可以去看代码)

  然后,这道题带了修改(其实并没有什么不同,只是需要修改块的大小以及多一个时间流逝的操作),于是给每个操作多一个时间戳即可。树上莫队和序列莫队最大的不同就在于如何转移。也就是我们如何由路径$(u_i,v_j)$走到路径$(u_j,v_j)$上去。其实就是先把$u_i$走到$u_j$,再把$v_i$走到$v_j$即可。注意,这里为了方便考虑,上述路径均没有把深度最小的点给算进去。这里没有讲清楚……还是往后看吧……

  至于证明吗……我们先设几个东西(下列内容参考vfleaking大神的博客)……

  设$S(u,v)$表示路径$(u,v)$上所有点的集合,$T(u,v)$表示$S(u,v)$去掉$lca(u,v)$之后剩下的点集。那么显然有:$$T(u,v)=S(root,u) \  xor \  S(root,v)$$

  这里的$xor$指的是集合的对称差。简单来说就是把出现偶数次的节点给删掉。这个操作显然和异或差不多,性质也差不多。

  于是我们可以考虑一下从$T(u_i,v_i)$转移到$T(u_j,v_j)$需要改变一些什么。$$\because T(u_i,v_i)=S(root,u_i)\ xor \ S(root,v_i)\quad T(u_j,v_j)=S(root,u_j)\ xor \ S(root,v_j)$$ $$\therefore T(u_i,v_i)\ xor \ T(u_j,v_j)=T(u_i,u_j)\ xor \ T(v_i,v_j)$$ $$\therefore T(u_j,v_j)=T(u_i,v_i)\ xor \ T(u_i,u_j)\ xor \ T(v_i,v_j)$$

  于是我们从$T(u_i,v_i)$转移到$T(u_j,v_j)$只需要把点集$T(u_i,u_j)$和$T(v_i,v_j)$中的点存在性全部取反即可。

  至于如何$O(1)$地进行插入和删除就不需要我多说了吧……

  其实我感觉有vfleaking大神的博客应该不需要我的这一篇才对

  下面贴代码:

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#define File(s) freopen(s".in","r",stdin),freopen(s".out","w",stdout)
#define maxn 100010 using namespace std;
typedef long long llg; int n,m,q,V[maxn],co[maxn],num[maxn];
int fa[maxn][17],dep[maxn],siz[maxn];
int head[maxn],next[maxn<<1],to[maxn<<1],tt;
int qt[maxn],fr[maxn],qto,nblo,sizblo;
llg sw[maxn],now,ans[maxn]; bool vis[maxn];
struct data{
int u,v,t;
bool operator < (const data &h)const{
if(fr[u]!=fr[h.u]) return fr[u]<fr[h.u];
if(fr[v]!=fr[h.v]) return fr[v]<fr[h.v];
return t<h.t;
}
}s[maxn],cha[maxn<<1]; int getint(){
int w=0;bool q=0;
char c=getchar();
while((c>'9'||c<'0')&&c!='-') c=getchar();
if(c=='-') c=getchar(),q=1;
while(c>='0'&&c<='9') w=w*10+c-'0',c=getchar();
return q?-w:w;
} void link(int x,int y){
to[++tt]=y;next[tt]=head[x];head[x]=tt;
to[++tt]=x;next[tt]=head[y];head[y]=tt;
} void dfs(int u,int ff){
qt[++qto]=u;
fa[u][0]=ff; dep[u]=dep[ff]+1;
for(int i=1,now=2;now<dep[u];i++,now<<=1)
fa[u][i]=fa[fa[u][i-1]][i-1];
for(int i=head[u],v;v=to[i],i;i=next[i])
if(v!=ff){
dfs(v,u); siz[u]+=siz[v];
if(siz[u]>=sizblo){
nblo++; siz[u]=0;
while(qt[qto]!=u) fr[qt[qto--]]=nblo;
}
}
siz[u]++;
} void change(int u){
now-=V[co[u]]*sw[num[co[u]]];
if(vis[u]) num[co[u]]--,vis[u]=0;
else num[co[u]]++,vis[u]=1;
now+=V[co[u]]*sw[num[co[u]]];
} void timego(int x){
if(!cha[x].t) return;
bool ww=vis[cha[x].t];
if(ww) change(cha[x].t);
co[cha[x].t]=cha[x].v;
if(ww) change(cha[x].t);
} int lca(int u,int v){
if(dep[u]<dep[v]) swap(u,v); int t=0;
for(int now=1;now<dep[u];now<<=1) t++;t--;
for(int i=t;i>=0;i--)
if(dep[fa[u][i]]>=dep[v]) u=fa[u][i];
if(u==v) return u;
for(int i=t;i>=0;i--)
if(fa[u][i]!=fa[v][i])
u=fa[u][i],v=fa[v][i];
return fa[u][0];
} void query(int u,int v){
int g=lca(u,v);
while(u!=g) change(u),u=fa[u][0];
while(v!=g) change(v),v=fa[v][0];
} int main(){
File("a");
n=getint(); m=getint(); q=getint();
for(int i=1;i<=m;i++) V[i]=getint();
for(int i=1;i<=n;i++) sw[i]=sw[i-1]+getint();
for(int i=1;i<n;i++) link(getint(),getint());
sizblo=pow(n,2.0/3.0)+1; dfs(1,0); tt=0;
for(int i=1;i<=n;i++) if(!fr[i]) fr[i]=nblo;
for(int i=1;i<=n;i++) co[i]=num[i]=getint();
for(int i=1,ty;i<=q;i++){
ty=getint(); s[i].t=++tt;
s[i].u=getint(); s[i].v=getint();
if(!ty){
cha[i].u=num[s[i].u]; cha[i].t=s[i].u;
cha[i].v=num[s[i].u]=s[i].v;
cha[i+q]=cha[i]; swap(cha[i+q].u,cha[i+q].v);
}
else if(fr[s[i].u]>fr[s[i].v]) swap(s[i].u,s[i].v);
}
for(int i=1;i<=n;i++) num[i]=0;
sort(s+1,s+q+1);
for(int i=1,nt=0,u=1,v=1,g;i<=q;i++){
if(cha[s[i].t].t) continue;
g=lca(s[i].u,s[i].v);
while(nt<s[i].t) timego(++nt);
while(nt>s[i].t) timego(q+(nt--));
query(u,s[i].u); query(v,s[i].v);
change(g); ans[s[i].t]=now; change(g);
u=s[i].u; v=s[i].v;
}
for(int i=1;i<=q;i++)
if(!cha[i].t) printf("%lld\n",ans[i]);
return 0;
}

UOJ #58 【WC2013】 糖果公园的更多相关文章

  1. [BZOJ3052][UOJ#58][WC2013]糖果公园

    [BZOJ3052][UOJ#58][WC2013]糖果公园 试题描述 Candyland 有一座糖果公园,公园里不仅有美丽的风景.好玩的游乐项目,还有许多免费糖果的发放点,这引来了许多贪吃的小朋友来 ...

  2. BZOJ3052/UOJ#58 [wc2013]糖果公园 莫队 带修莫队 树上莫队

    原文链接https://www.cnblogs.com/zhouzhendong/p/BZOJ3052.html 题目传送门 - BZOJ3052 题目传送门 - UOJ#58 题意 给定一棵树,有 ...

  3. bzoj 3052: [wc2013]糖果公园 带修改莫队

    3052: [wc2013]糖果公园 Time Limit: 250 Sec  Memory Limit: 512 MBSubmit: 506  Solved: 189[Submit][Status] ...

  4. 洛谷 P4074 [WC2013]糖果公园 解题报告

    P4074 [WC2013]糖果公园 糖果公园 树上待修莫队 注意一个思想,dfn序处理链的方法,必须可以根据类似异或的东西,然后根据lca分两种情况讨论 注意细节 Code: #include &l ...

  5. AC日记——[WC2013]糖果公园 cogs 1817

    [WC2013]糖果公园 思路: 带修改树上莫队(模板): 来,上代码: #include <cmath> #include <cstdio> #include <cst ...

  6. COGS1817. [WC2013]糖果公园

    1817. [WC2013]糖果公园 ★★★☆   输入文件:park.in   输出文件:park.out   简单对比时间限制:8 s   内存限制:512 MB [题目描述] Candyland ...

  7. 【BZOJ3052】[wc2013]糖果公园 带修改的树上莫队

    [BZOJ3052][wc2013]糖果公园 Description Input Output Sample Input Sample Input Sample Output 84 131 27 84 ...

  8. 【BZOJ】3052: [wc2013]糖果公园 树分块+带修改莫队算法

    [题目]#58. [WC2013]糖果公园 [题意]给定n个点的树,m种糖果,每个点有糖果ci.给定n个数wi和m个数vi,第i颗糖果第j次品尝的价值是v(i)*w(j).q次询问一条链上每个点价值的 ...

  9. 【Luogu P4074】[WC2013]糖果公园(树上带修改莫队)

    题目描述 Candyland 有一座糖果公园,公园里不仅有美丽的风景.好玩的游乐项目,还有许多免费糖果的发放点,这引来了许多贪吃的小朋友来糖果公园游玩. 糖果公园的结构十分奇特,它由 \(n\) 个游 ...

  10. BZOJ.3052.[WC2013]糖果公园(树上莫队 带修改莫队)

    题目链接 BZOJ 当然哪都能交(都比在BZOJ交好),比如UOJ #58 //67376kb 27280ms //树上莫队+带修改莫队 模板题 #include <cmath> #inc ...

随机推荐

  1. AJAX(一)

    AJAX(一) Ajax是Asynchronous Javascript和XML的简写,这一技术能够向服务器请求额外的数据而无需卸载页面,会带来更好的用户体验. [前面的基础知识][关于同步和异步的了 ...

  2. 定向爬虫 - Python模拟新浪微博登录

    当我们试图从新浪微博抓取数据时,我们会发现网页上提示未登录,无法查看其他用户的信息. 模拟登录是定向爬虫制作中一个必须克服的问题,只有这样才能爬取到更多的内容. 实现微博登录的方法有很多,一般我们在模 ...

  3. cookie保存中文登录账号获取时乱码问题

    登录成功后写入cookie的代码 Response.Cookies["account"].Value = account;//"管理员" Response.Co ...

  4. Entity Framework 教程——什么是Entity Framework

    什么是Entity Framework 编写和管理ADO.NET是一个繁琐而又无聊的工作.微软为你的应用提供了一个名为"Entity Framework"的ORM框架来自动化管理你 ...

  5. C#、JAVA操作Hadoop(HDFS、Map/Reduce)真实过程概述。组件、源码下载。无法解决:Response status code does not indicate success: 500。

    一.Hadoop环境配置概述 三台虚拟机,操作系统为:Ubuntu 16.04. Hadoop版本:2.7.2 NameNode:192.168.72.132 DataNode:192.168.72. ...

  6. redis开启远程访问

    redis默认只允许本地访问,要使redis可以远程访问可以修改redis.conf   打开redis.conf文件在NETWORK部分有说明   ######################### ...

  7. 在Wcf中应用ProtoBuf替代默认的序列化器

    Google的ProtoBuf序列化器性能的牛逼已经有目共睹了,可以把它应用到Socket通讯,队列,Wcf中,身为dotnet程序员一边期待着不久后Grpc对dotnet core的支持更期待着Wc ...

  8. 谈一谈前端多容器(多webview平台)处理方案

    文中是我个人的一些开发经验,希望对各位有用,也希望各位多多支持讨论,指出文中不足以及提出您的一些建议. 双容器 得益于近几年移动端的发展,前端早已今非昔比,从大型框架来说angularJS.react ...

  9. C++实现DNS域名解析

    一.概述 现在来搞定DNS域名解析,其实这是前面一篇文章C++实现Ping里面的遗留问题,要干的活是ping的过程中画红线的部分: cmd下域名解析的命令是nslookup,比如“nslookup w ...

  10. (视频)Erich Gamma 与 Visual Studio Online 的一点野史

    大家对Erich Gamma的熟悉应该都集中在<设计模式>, Eclipse以及最近微软所发布的Visual Studio Code跨平台代码编辑器.其实在Erich加入微软的头几年里,他 ...