树上莫队和普通的序列莫队很像,我们把树进行dfs,然后存一个长度为2n的括号序列,就是一个点进去当作左括号,出来当作右括号,然后如果访问从u到v路径,我们可以转化成括号序列的区间,记录x进去的时候编号为f[x],出来时为g[x],然后分类讨论一下(f[u]<f[v]),如果u和v的lca不是u,那么就是从g[u]到f[v],否则就是lca的f到另一个点的f,(可以自己试一下,中间过程没有用的点正好就抵消掉了)这里要注意一下,从g[u]到f[v]的时候我们会少掉lca这个点,特殊处理一下即可,然后按照普通莫队排一下序,暴力就行了。 —— by VANE

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N=;
int n,m,cnt1,cnt2,tot,clk,f[N],g[N];
vector<int> M[N];
int id[N<<],blg[N<<];
int bin[],pos[N],fa[N][],c[N],d[N];
int v[N],w[N],last[N],u[N];
bool vis[N];
struct node
{
int l,r,t,id;
}a[N],b[N];
ll ans[N],sum;
void dfs(int x)
{
f[x]=++clk;id[clk]=x;
for(int i=;bin[i]<=d[x];++i)
fa[x][i]=fa[fa[x][i-]][i-];
for(int i=;i<M[x].size();++i)
{
int y=M[x][i];
if(y!=fa[x][])
{
fa[y][]=x;
d[y]=d[x]+;
dfs(y);
}
}
g[x]=++clk;
id[clk]=x;
}
int lca(int x,int y)
{
if(d[x]<d[y]) swap(x,y);
int tmp=d[x]-d[y];
for(int i=;bin[i]<=tmp;++i)
if(tmp&bin[i]) x=fa[x][i];
if(x==y) return x;
for(int i=;i>=;--i)
if(fa[x][i]!=fa[y][i])
x=fa[x][i],y=fa[y][i];
return fa[x][];
}
bool cmp(node x,node y)
{
if(blg[x.l]<blg[y.l]) return ;
if(blg[x.l]==blg[y.l]&&blg[x.r]<blg[y.r]) return ;
if(blg[x.l]==blg[y.l]&&blg[x.r]==blg[y.r]) return x.t<y.t;
return ;
}
void modify(int x)
{
if(vis[x]) sum-=1ll*v[c[x]]*w[u[c[x]]--];
else sum+=1ll*v[c[x]]*w[++u[c[x]]];
vis[x]^=;
}
void change(int x,int y)
{
if(vis[x]) {modify(x);c[x]=y;modify(x);}
else c[x]=y;
}
int main()
{
int cas;
scanf("%d%d%d",&n,&m,&cas);
bin[]=;for(int i=;i<=;++i) bin[i]=bin[i-]<<;
for(int i=;i<=m;++i) scanf("%d",v+i);
for(int i=;i<=n;++i) scanf("%d",w+i);
for(int i=;i<n;++i)
{
int l,r;scanf("%d%d",&l,&r);
M[l].push_back(r);
M[r].push_back(l);
}
for(int i=;i<=n;++i)
scanf("%d",c+i),last[i]=c[i];
int sz=pow(n,2.0/);
dfs();
for(int i=;i<=clk;++i) blg[i]=(i-)/sz;
while(cas--)
{
int l,r,t;
scanf("%d%d%d",&t,&l,&r);
if(t)
{
if(f[l]>f[r]) swap(l,r);
a[++cnt1].r=f[r];a[cnt1].t=cnt2;
a[cnt1].id=cnt1;
a[cnt1].l=(lca(l,r)==l)?f[l]:g[l];
}
else
{
b[++cnt2].l=l;b[cnt2].t=last[l];
last[l]=b[cnt2].r=r;
}
}
sort(a+,a++cnt1,cmp);
int l=,r=,t=;
for(int i=;i<=cnt1;++i)
{
for(;t<=a[i].t;++t) change(b[t].l,b[t].r);
for(;t>a[i].t;--t) change(b[t].l,b[t].t);
while(l>a[i].l) modify(id[--l]);
while(l<a[i].l) modify(id[l++]);
while(r>a[i].r) modify(id[r--]);
while(r<a[i].r) modify(id[++r]);
int x=id[l],y=id[r],tmp=lca(x,y);
if(x!=tmp&&y!=tmp) {modify(tmp);ans[a[i].id]=sum;modify(tmp);}
else ans[a[i].id]=sum;
}
for(int i=;i<=cnt1;++i)
printf("%lld\n",ans[i]);
}

BZOJ3052 [wc2013] 糖果公园 【树上莫队】的更多相关文章

  1. BZOJ3052:[WC2013]糖果公园(树上莫队)

    Description Input Output Sample Input 4 3 51 9 27 6 5 12 33 13 41 2 3 21 1 21 4 20 2 11 1 21 4 2 Sam ...

  2. P4074 [WC2013]糖果公园 树上莫队带修改

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

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

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

  4. BZOJ 3052: [wc2013]糖果公园 | 树上莫队

    题目: UOJ也能评测 题解 请看代码 #include<cstdio> #include<algorithm> #include<cstring> #includ ...

  5. 【WC2013】 糖果公园 - 树上莫队

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

  6. 【WC2013】糖果公园 [树上莫队]

    题意: 一棵树,修改一个点的颜色,询问两点路径上每种颜色的权值$val[c]$*出现次数的权值$cou[w[c]]$的和 sro VFK 树上莫队 按照王室联邦的方法分块,块的大小直径个数有保证,并不 ...

  7. 洛谷P4074 [WC2013]糖果公园(莫队)

    传送门 总算会树形莫队了…… 上次听说树形莫队是给树分块,实在看不懂.然后用括号序列的方法做总算能弄明白了 先说一下什么是括号序列,就是在$dfs$的时候,进入的时候记录一下,出去的时候也记录一下 拿 ...

  8. 【BZOJ-3052】糖果公园 树上带修莫队算法

    3052: [wc2013]糖果公园 Time Limit: 200 Sec  Memory Limit: 512 MBSubmit: 883  Solved: 419[Submit][Status] ...

  9. LUOGU P4074 [WC2013]糖果公园 (树上带修莫队)

    传送门 解题思路 树上带修莫队,搞了两天..终于开O2+卡常大法贴边过了...bzoj上跑了183s..其实就是把树上莫队和带修莫队结合到一起,首先求出括号序,就是进一次出一次那种的,然后如果求两个点 ...

随机推荐

  1. 无废话JavaScript(下)

    五.函数式 这个可不是JavaScript的发明,它的发明人已经死了,而他的这个发明还在困扰着我们……如同爱迪生的灯泡还在照耀着我们. 其实函数式语言很简单,它就是一种与命令式语言同样“完备”的语言实 ...

  2. 【BZOJ】3302: [Shoi2005]树的双中心 && 2103: Fire 消防站 && 2447: 消防站

    [题意]给定带点权树,要求选择两个点x,y,满足所有点到这两个点中较近者的距离*点权的和最小.n<=50000,h<=100. [算法]树的重心 [题解]代码参考自:cgh_Andy 观察 ...

  3. 1030 大数进制转换(51Nod + JAVA)

    题目链接:https://www.51nod.com/onlineJudge/questionCode.html#!problemId=1030 题目: 代码实现如下: import java.mat ...

  4. python操作YAML文件之pyyaml库

    1. YAML简介 YAML是一种被认为可以超越XML.JSON的配置文件,最早接触是Spring Boot,木有想到python也是支持的,遂研究一下. python解析YAML库叫做pyyaml, ...

  5. 48、面向对象中super的作用?

    什么是super? super() 函数是用于调用父类(超类)的一个方法. super 是用来解决多重继承问题的,直接用类名调用父类方法在使用单继承的时候没问题,但是如果使用多继承,会涉及到查找顺序( ...

  6. bzoj 3083 树链剖分

    首先我们先将树提出一个根变成有根树,那么我们可以通过树链剖分来实现对于子树的最小值求解,那么按照当前的根和询问的点的相对位置关系我们可以将询问变成某个子树和或者除去某颗子树之后其余的和,前者直接询问区 ...

  7. Android: 在onCreate()中获得对象尺寸

    onCreate() 中 View 尚未绘制完成 很多时候,我们需要在某个界面刚刚加载的时候,执行一些对 View 进行操作的代码,通常我们把这些代码放在 Activity 的 onCreate() ...

  8. rds 与mysql 进行主从同步

    .rds上默认会有server-****,只需要配置从数据库: .从数据库的配置流程: .[mysqld] log-bin = mysql-bin-changelog #要和主库中的名字一样 rela ...

  9. explicit浅谈

    在C++中,explicit关键字主要用于防止隐式转换,用于修饰构造函数.复制构造函数. 例如有一个类: class A { public: A( int count ) : m_data( coun ...

  10. EasyUi – 6.easyui常见问题

    1.进度条 2.JQuery EasyUI弹出对话框解决Asp.net服务器控件无法执行后台代码的方法 3. 三张表的连接查询现在到datagrid里 4.日期组合框DateBox设置readonly ...