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

【题目】#58. 【WC2013】糖果公园

【题意】给定n个点的树,m种糖果,每个点有糖果ci。给定n个数wi和m个数vi,第i颗糖果第j次品尝的价值是v(i)*w(j)。q次询问一条链上每个点价值的和或修改一个点的糖果ci。n,m,q<=10^5。

【算法】树分块+带修改莫队算法

【题解】参考:WC 2013 糖果公园 park 题解 by vfleaking

首先树分块,参考王室联邦的方法。确定块大小为B,一遍DFS可以分成若干大小为[B,3B]的块,性质是块内两点距离至多为B。

定义(x,y,t)表示询问经过了t次修改的树链x-y的答案,将询问排序:第一关键字belong[x],第二关键字belong[y],第三关键字t。

对于一个询问,要考虑从上一次询问(x‘,y‘,t‘)转移。首先转移t,只需要记录每次修改前和修改后的数值,就可以实现修改或逆修改了。

然后是从树链x‘-y‘转移到树链x-y‘,这里需要异或操作(对称差)。所谓异或操作,就是如果x标记过就减去答案并消除标记,如果x没标记过就加上答案并增加标记。

具体过程可以参考vfk的公式推导,感性理解也很简单:定义t(x,y)表示除了lca(x,y)的树链x-y,那么 t(x,y‘) = t(x‘,y‘) ^ t(x,x‘)

有了这个,我们只要在当前基础上异或一下t(x,x‘)和t(y,y‘)就可以实现从x‘-y‘转移到x-y了,当然LCA全部另外算就可以了。

另外,之前修改点x的数值时,如果在当前答案中必须消除,修改,再加入。

【复杂度分析】假设块大小为B。(随便看看就好了,这部分不保证正确……)

如果u和v都不移出块,那么位置移动复杂度O(q*B),时间移动复杂度O(q)。

如果v移出块,那么因为belong[u]只有n/B种可能,位置移动复杂度O(n*n/B)。

如果u移出块,那么位置移动的复杂度O(n)。

而belong[u],belong[v]只有(n/B)^2种可能,所以时间移动的复杂度是O(q*(n/B)^2)。

平衡后B=n^(2/3)。

所以总复杂度O(n^(5/3))。

因为树分块的块大小实际上比B大,所以取B=N^(2/3)*0.5时常数比较优秀。

#include<cmath>
#include<cstdio>
#include<cctype>
#include<cstring>
#include<algorithm>
using namespace std;
int read(){
    int s=0,t=1;char c;
    while(!isdigit(c=getchar()))if(c==‘-‘)t=-1;
    do{s=s*10+c-‘0‘;}while(isdigit(c=getchar()));
    return s*t;
}
const int maxn=100010;
int tot,n,m,q,B,top,cnt,c0,c1;
int first[maxn],deep[maxn],f[maxn][20],belong[maxn],st[maxn],num[maxn],c[maxn],w[maxn],v[maxn],pre[maxn];
long long ans,ANS[maxn];
bool vis[maxn];
struct edge{int v,from;}e[maxn*2];
struct C0{int x,y,pre;}a[maxn];
struct C1{int x,y,t,id;}b[maxn];
void insert(int u,int v){tot++;e[tot].v=v;e[tot].from=first[u];first[u]=tot;}
void dfs(int x,int fa){
    int lim=top;
    for(int j=1;(1<<j)<=deep[x];j++)f[x][j]=f[f[x][j-1]][j-1];
    for(int i=first[x];i;i=e[i].from)if(e[i].v!=fa){
        deep[e[i].v]=deep[x]+1;
        f[e[i].v][0]=x;
        dfs(e[i].v,x);
        if(top-lim>=B){
            cnt++;
            while(top>lim)belong[st[top--]]=cnt;
        }
    }
    st[++top]=x;
}
int lca(int x,int y){
    if(deep[x]<deep[y])swap(x,y);
    int d=deep[x]-deep[y];
    for(int j=0;(1<<j)<=d;j++)if((1<<j)&d)x=f[x][j];
    if(x==y)return x;
    for(int j=17;j>=0;j--)if((1<<j)<=deep[x]&&f[x][j]!=f[y][j])x=f[x][j],y=f[y][j];
    return f[x][0];
}
void reverse(int x){
    if(vis[x])ans-=1ll*w[num[c[x]]--]*v[c[x]];
    else ans+=1ll*w[++num[c[x]]]*v[c[x]];
    vis[x]^=1;
}
void modify(int x,int y){
    if(!vis[x])c[x]=y;
    else reverse(x),c[x]=y,reverse(x);
}
void solve(int x,int y){
    while(x!=y){
        if(deep[x]>deep[y])reverse(x),x=f[x][0];
        else reverse(y),y=f[y][0];
    }
}
bool cmp(C1 a,C1 b){return belong[a.x]<belong[b.x]||(belong[a.x]==belong[b.x]&&belong[a.y]<belong[b.y])||
(belong[a.x]==belong[b.x]&&belong[a.y]==belong[b.y]&&a.t<b.t);}

int main(){
    n=read();m=read();q=read();
    for(int i=1;i<=m;i++)v[i]=read();
    for(int i=1;i<=n;i++)w[i]=read();
    for(int i=1;i<n;i++){
        int u=read(),v=read();
        insert(u,v);insert(v,u);
    }
    B=pow(n,2.0/3)*0.5;
    dfs(1,0);
    while(top)belong[st[top--]]=cnt;
    for(int i=1;i<=n;i++)c[i]=pre[i]=read();
    for(int i=1;i<=q;i++){
        int kind=read(),x=read(),y=read();
        if(!kind){
            a[++c0]=(C0){x,y,pre[x]},pre[x]=y;
        }
        else{
            b[++c1]=(C1){x,y,c0,c1};
        }
    }
    sort(b+1,b+c1+1,cmp);
    for(int i=1;i<=b[1].t;i++)modify(a[i].x,a[i].y);
    solve(b[1].x,b[1].y);
    int c=lca(b[1].x,b[1].y);
    reverse(c);ANS[b[1].id]=ans;reverse(c);
    for(int i=2;i<=c1;i++){
        for(int j=b[i-1].t+1;j<=b[i].t;j++)modify(a[j].x,a[j].y);
        for(int j=b[i-1].t;j>b[i].t;j--)modify(a[j].x,a[j].pre);
        solve(b[i-1].x,b[i].x);solve(b[i-1].y,b[i].y);
        int c=lca(b[i].x,b[i].y);
        reverse(c);ANS[b[i].id]=ans;reverse(c);
    }
    for(int i=1;i<=c1;i++)printf("%lld\n",ANS[i]);
    return 0;
}

原文地址:https://www.cnblogs.com/onioncyc/p/8573121.html

时间: 2024-08-03 17:58:23

【BZOJ】3052: [wc2013]糖果公园 树分块+待修改莫队算法的相关文章

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

题目描述 Candyland 有一座糖果公园,公园里不仅有美丽的风景.好玩的游乐项目,还有许多免费糖果的发放点,这引来了许多贪吃的小朋友来糖果公园游玩. 糖果公园的结构十分奇特,它由 \(n\) 个游览点构成,每个游览点都有一个糖果发放处,我们可以依次将游览点编号为 \(1\) 至 \(n\).有 \(n-1\) 条双向道路连接着这些游览点,并且整个糖果公园都是连通的,即从任何一个游览点出发都可以通过这些道路到达公园里的所有其它游览点. 糖果公园所发放的糖果种类非常丰富,总共有 \(m\) 种,

[UOJ #58][WC2013]糖果公园(树上带修改莫队)

Description Solution 树上带修改莫队…!VFK的题解写得很清楚啦 (我的程序为什么跑得这么慢…交的时候总有一种自己在卡测评的感觉…) #include<iostream> #include<cstdio> #include<cstdlib> #include<cstring> #include<cmath> #include<algorithm> #define MAXN 100005 typedef long l

[bzoj 3052][wc2013]糖果公园

传送门:http://www.lydsy.com/JudgeOnline/problem.php?id=3052 [wc2013]糖果公园 Time Limit: 200 Sec  Memory Limit: 512 MBSubmit: 1213  Solved: 609[Submit][Status][Discuss] Description Input Output Sample Input Sample Input Sample Output 84 131 27 84 树上莫队,把树分块,

bzoj 3052: [wc2013]糖果公园(带修改的树上莫队)

3052: [wc2013]糖果公园 Time Limit: 200 Sec  Memory Limit: 512 MB Submit: 892  Solved: 425 [Submit][Status][Discuss] Description Input Output Sample Input Sample Input Sample Output 84 131 27 84 HINT Source [Submit][Status][Discuss] 题解:bzoj 2120 和 bzoj 37

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

3052: [wc2013]糖果公园 Time Limit: 250 Sec  Memory Limit: 512 MBSubmit: 506  Solved: 189[Submit][Status] Description Input Output Sample Input Sample Input Sample Output 84 131 27 84 HINT 本来这道题想到了莫队算法,但是看到带修改就直接放弃了.结果看题解才发现带修改居然也能用莫队!!!之所以可以这样用,是因为修改的时间复

BZOJ 3052 WC2013 糖果公园 带修改树上莫队

题目大意:给定一棵树,每个点有一个颜色,提供两种操作: 1.询问两点间路径上的Σv[a[i]]*w[k],其中a[i]代表这个点的颜色,k表示这个点是这种颜色第k次出现 2.修改某个点的颜色 VfleaKing的题解见 http://vfleaking.blog.163.com/blog/static/174807634201311011201627/ 带修改莫队上树--如果不带修改就正常搞就好了 如果带修改的话,令块的大小为n^(2/3) 将询问排序时第一键值为左端点所在块,第二键值为右端点所

[BZOJ 3052] [wc2013] 糖果公园 【树上莫队】

题目链接:BZOJ - 3052 题目分析 这道题就是非常经典的树上莫队了,并且是带修改的莫队. 带修改的莫队:将询问按照 左端点所在的块编号为第一关键字,右端点所在的块为第二关键字,位于第几次修改之后为第三关键字 排序. 我们将块的大小设置在 n^(2/3) ,这样一共有 n^(1/3) 个块.最后算法的总复杂度会是 n^(5/3) . 每一次如何从上一个询问转移来呢? 假设上一个询问是 (lx, ly, lt) ,这次的询问是 (x, y, t) .t 代表是在第 t 次修改操作之后. 首先

BZOJ 3052 [wc2013]糖果公园 树上莫队

题意:链接 方法:莫队上树 解析: 首先先考虑个莫队的算法,树分块 然后怎么做呢? 可以设两个指针么,然后按一定的排序方式移动指针,使复杂度可过. 观察到这道题是100s的,n^2是GG的,但比n^2稍微小一点就不GG了, 所以排序的作用就体现了. 按照左端点所在块为第一关键字,右端点所在块为第二关键字出现时间为第三关键字排序. 一共有n的1/3次方个块,所以所有的询问的左右端点所在的块的取值最多之后n的2/3次方种,对于每一种,对时间的修改最多会从尾改到头,所以又多乘一个q,n,q同阶,所以总

[BZOJ 4129]Haruna’s Breakfast(树上带修改莫队)

Description Haruna每天都会给提督做早餐! 这天她发现早饭的食材被调皮的 Shimakaze放到了一棵 树上,每个结点都有一样食材,Shimakaze要考验一下她. 每个食材都有一个美味度,Shimakaze会进行两种操作: 1.修改某个结点的食材的美味度. 2.对于某条链,询问这条链的美味度集合中,最小的未出现的自然数是多少.即mex值. 请你帮帮Haruna吧. Solution 树上带修改莫队 统计答案的时候也分块查询,找到第一个没满的块开始一个一个找 #include<i