题意:给定一个n个点的树,每个结点上有一种颜色c[i]

定义一条简单路径的偷税指数为simga (sigma w[i](i=0..a[j]))*v[j](j=0..m),其中a[i]为第i种颜色在路径上出现的次数

现在共有q个非强制在线的操作,格式为(op,x,y)

op=0时代表将x号点的颜色修改为y

op=1时询问(x,y)这条简单路径的偷税指数

思路:

 #include<bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef unsigned int uint;
typedef unsigned long long ull;
typedef pair<int,int> PII;
typedef pair<ll,ll> Pll;
typedef vector<int> VI;
typedef vector<PII> VII;
//typedef pair<ll,ll>P;
#define N 200010
#define M 200010
#define fi first
#define se second
#define MP make_pair
#define pb push_back
#define pi acos(-1)
#define mem(a,b) memset(a,b,sizeof(a))
#define rep(i,a,b) for(int i=(int)a;i<=(int)b;i++)
#define per(i,a,b) for(int i=(int)a;i>=(int)b;i--)
#define lowbit(x) x&(-x)
#define Rand (rand()*(1<<16)+rand())
#define id(x) ((x)<=B?(x):m-n/(x)+1)
#define ls p<<1
#define rs p<<1|1 const int MOD=1e9+,inv2=(MOD+)/;
double eps=1e-;
int INF=0x7fffffff;
int inf=1e9;
int dx[]={-,,,};
int dy[]={,,-,}; struct Q
{
int l,r,cur,id;
}q[M]; struct P
{
int x,pre,now;
}p[M]; int head[N],vet[N],nxt[N],w[N],c[N],cnt[N],tot,
f[N][],l[N],r[N],pos[N],dfn[N],to[N],dep[N],v[N],vis[N],b[N],tim1,tim2;
ll now,ans[N]; bool cmp(Q a,Q b)
{
return pos[a.l]<pos[b.l]
||pos[a.l]==pos[b.l]&&pos[a.r]<pos[b.r]
||pos[a.l]==pos[b.l]&&pos[a.r]==pos[b.r]&&a.cur<b.cur;
} int read()
{
int v=,f=;
char c=getchar();
while(c<||<c) {if(c=='-') f=-; c=getchar();}
while(<=c&&c<=) v=(v<<)+v+v+c-,c=getchar();
return v*f;
} void add(int a,int b)
{
nxt[++tot]=head[a];
vet[tot]=b;
head[a]=tot;
} void dfs(int u,int fa)
{
rep(i,,) f[u][i]=f[f[u][i-]][i-];
dfn[u]=++tim1;
l[u]=++tim2; to[tim2]=u;
int e=head[u];
while(e)
{
int v=vet[e];
if(v!=fa)
{
dep[v]=dep[u]+;
f[v][]=u;
dfs(v,u);
}
e=nxt[e];
}
r[u]=++tim2; to[tim2]=u;
} int lca(int x,int y)
{
if(dep[x]<dep[y]) swap(x,y);
int d=dep[x]-dep[y];
rep(i,,)
if(d>>i&) x=f[x][i];
per(i,,)
if(f[x][i]!=f[y][i])
{
x=f[x][i];
y=f[y][i];
}
if(x==y) return x;
return f[x][];
} void go(int x)
{
if(vis[x])
{
now-=1ll*v[c[x]]*w[cnt[c[x]]];
cnt[c[x]]--;
}
else
{
cnt[c[x]]++;
now+=1ll*v[c[x]]*w[cnt[c[x]]];
}
vis[x]^=;
} void update(int x,int y)
{
if(vis[x])
{
go(x);
c[x]=y;
go(x);
}
else c[x]=y;
} int main()
{
int n=read(),m=read(),k=read();
int S=max(,(int)pow(n,2.0/));
rep(i,,m) v[i]=read();
rep(i,,n) w[i]=read();
tot=;
rep(i,,n) head[i]=;
rep(i,,n-)
{
int x=read(),y=read();
add(x,y);
add(y,x);
}
rep(i,,n)
{
c[i]=read();
b[i]=c[i];
} tim1=tim2=;
dfs(,);
rep(i,,tim2) pos[i]=(i-)/S;
int l1=,l2=;
rep(i,,k)
{
int op=read(),x=read(),y=read();
if(op==)
{
l2++;
p[l2].x=x;
p[l2].pre=b[x];
p[l2].now=b[x]=y;
}
else
{
l1++;
if(dfn[x]>dfn[y]) swap(x,y);
int t=lca(x,y);
if(t==x) q[l1].l=l[x];
else q[l1].l=r[x];
q[l1].r=l[y];
q[l1].id=l1;
q[l1].cur=l2;
}
}
sort(q+,q+l1+,cmp);
int L=,R=,T=;
now=;
rep(i,,l1)
{
while(T<q[i].cur)
{
T++;
update(p[T].x,p[T].now);
}
while(T>q[i].cur)
{
update(p[T].x,p[T].pre);
T--;
}
while(L>q[i].l) go(to[--L]);
while(L<q[i].l) go(to[L++]);
while(R>q[i].r) go(to[R--]);
while(R<q[i].r) go(to[++R]);
int x=to[L],y=to[R],t=lca(x,y);
if(x!=t&&y!=t) go(t);
ans[q[i].id]=now;
if(x!=t&&y!=t) go(t);
}
rep(i,,l1) printf("%lld\n",ans[i]);
return ;
}

【BZOJ3052&UOJ58】糖果公园(树上带修莫队)的更多相关文章

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

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

  2. luogu4074 [WC2013]糖果公园(树上带修莫队)

    link 题目大意:给一个树,树上每个点都有一种颜色,每个颜色都有一个收益 每次修改一个点上的颜色 或询问一条链上所有颜色第i次遇到颜色j可以获得w[i]*v[j]的价值,求链上价值和 题解:树上带修 ...

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

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

  4. [WC2013][luogu4074] 糖果公园 [树上带修改莫队]

    题面: 传送门 思路: 一道实现起来细节比较恶心的题目 但是其实就是一个裸的树上带修改莫队 好像树上莫队也出不了什么结合题目,不像序列莫队天天结合AC自动机.后缀数组...... 莫队学习请戳这里:莫 ...

  5. BZOJ3052: [wc2013]糖果公园【树上带修莫队】

    Description Input Output Sample Input Sample Input Sample Output 84 131 27 84 HINT 思路 非常模板的树上带修莫队 真的 ...

  6. BZOJ 3052/Luogu P4074 [wc2013]糖果公园 (树上带修莫队)

    题面 中文题面,难得解释了 BZOJ传送门 Luogu传送门 分析 树上带修莫队板子题... 开始没给分块大小赋初值T了好一会... CODE #include <bits/stdc++.h&g ...

  7. BZOJ 4129 Haruna’s Breakfast ( 树上带修莫队 )

    题面 求树上某路径上最小的没出现过的权值,有单点修改 添加链接描述 分析 树上带修莫队板题,问题是怎么求最小的没出现过的权值. 因为只有nnn个点,所以没出现过的最小值一定在[0,n][0,n][0, ...

  8. bzoj4129 Haruna’s Breakfast 树上带修莫队+分块

    题目传送门 https://lydsy.com/JudgeOnline/problem.php?id=4129 题解 考虑没有修改的序列上的版本应该怎么做: 弱化的题目应该是这样的: 给定一个序列,每 ...

  9. BZOJ 3052 树上带修莫队

    思路: 就是把带修莫队移到了树上 块的大小开到(n^2/3)/2 比较好- 这是一个卡OJ好题 //By SiriusRen #include <cmath> #include <c ...

  10. BZOJ3052(树上带修莫队)

    树上莫队的基本思路是把树按dfs序分块,然后先按x所在块从小到大排序,再按y所在块从小到大排序,处理询问即可. 这道题带修改,再加一个时间维即可. 设块大小为T,那么时间复杂度为$O(nT+\frac ...

随机推荐

  1. leetcode 118. 杨辉三角(python)

    给定一个非负整数 numRows,生成杨辉三角的前 numRows 行. 在杨辉三角中,每个数是它左上方和右上方的数的和. 示例: 输入: 5输出:[ [1], [1,1], [1,2,1], [1, ...

  2. scrapy电影天堂实战(二)创建爬虫项目

    公众号原文 创建数据库 我在上一篇笔记中已经创建了数据库,具体查看<scrapy电影天堂实战(一)创建数据库>,这篇笔记创建scrapy实例,先熟悉下要用到到xpath知识 用到的xpat ...

  3. Vagrant 手册之 Vagrantfile - 提示及技巧

    原文地址 Vagrantfile 是一种非常灵活的配置格式.语法基于 Ruby,可以用它做很多事情.在本页使用一些提示和技巧时,请注意正确使用它们. 1. 使用循环定义虚拟机 如果你想对多机器应用稍微 ...

  4. 2019牛客暑期多校训练营(第三场)H Magic Line

    原题链接:H  Magic Line 题意简述: 给定n个点,要求画一条直线将n个点分成均有n / 2个点的两部分,不能有点在线上: 解题思路: 首先,先将所有的点进行以x为第一关键字,y为第二关键字 ...

  5. CentOS7和Ubuntu18.10下运行Qt Creator出现cannot find -lGL的问题的解决方案

    解决方法:缺少相应的opengl的库,需要安装opengl库 一.Ubuntu下解决Qt5.11.1 cannot find -lGL 有两种原因: 一种是没有按照libGL库,那么就安装: sudo ...

  6. Manacher(最长回文串)

    http://acm.hdu.edu.cn/showproblem.php?pid=3068 最长回文 Problem Description 给出一个只由小写英文字符a,b,c...y,z组成的字符 ...

  7. Django文件上传下载与富文本编辑框

    django文件上传下载 上传 配置settings.py # 设定文件的访问路径,如:访问http://127.0.0.1:8000/media/就可以获取文件 MEDIA_URL = '/medi ...

  8. Pandas的高级操作

    pandas数据处理 1. 删除重复元素 使用duplicated()函数检测重复的行,返回元素为布尔类型的Series对象,每个元素对应一行,如果该行不是第一次出现,则元素为True keep参数: ...

  9. Codeforces 208E. Blood Cousins

    传送门 题目大意: 小C喜欢研究族谱,这一天小C拿到了一整张族谱. 小C先要定义一下k-祖先. x的1-祖先指的是x的父亲 x的k-祖先指的是x的(k-1)-祖先的父亲 小C接下来要定义k-兄弟 x的 ...

  10. nginx各版本全自动编译安装脚本

    #!/bin/bash #作者:星云法师(头条号:西西图图---专注美食领域的研究) #环境:centos7,如果是其它的系统可以相应做调整.#--------选择安装方式,网络晚装还是本地安装--- ...