题目大意:

给定一棵\(n\)个节点的带权树有根树,设\(sum_p\)表示以点\(p\)为根的这棵子树中所有节点的权

计算姬支持下列两种操作:

  1. 给定两个整数\(u,v\),修改点\(u\)的权值为\(v\)。
  2. 给定两个整数\(l,r\),计算\(\sum_{i=l}^rsum_i\)

题解:

表示自己没能想出来...被同桌嘲讽了QAQ...

首先是这道题的数据范围很奇怪,只有10W,这就说明了你有充足的时间来瞎搞

所以我们就瞎搞

如果没有修改操作那么我们直接\(O(n)\)预处理就可以\(O(1)\)询问了

但是我们存在修改操作,而且一次修改朴素是\(O(n)\)的

但是我们发现:无论我们要修改多少个数的值,修改的复杂度总是\(O(n)\)的

所以我们多攒几次修改在外部维护其对答案的影响,然后凑够了再一块改.

所以我们考虑如何外部维护.

首先我们将问题放到树的dfs序上,

我们发现:

这个问题实际上就是让我们统计一下当前询问的sum区间中有多少区间覆盖了我们修改的点

(注意,每一个sum实际上都代表了dfs序中的一个区间)

所以我们可以一次枚举每一个点,求一下当前的sum区间中有多少区间覆盖了当前枚举的点.

对于这个操作我们可以直接用可持久化线段树瞎搞一下就好了

经过本人的测试:在随机数据下,积累68个修改操作的时候进行修改,跑得最快 !

本题还有一个大坑点 : 答案会炸long long需要开unsigned long long 才能过...

#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
typedef long long ll;
typedef unsigned long long ull;
inline void read(int &x){
x=0;char ch;bool flag = false;
while(ch=getchar(),ch<'!');if(ch == '-') ch=getchar(),flag = true;
while(x=10*x+ch-'0',ch=getchar(),ch>'!');if(flag) x=-x;
}
inline void read(ll &x){
x=0;char ch;bool flag = false;
while(ch=getchar(),ch<'!');if(ch == '-') ch=getchar(),flag = true;
while(x=10*x+ch-'0',ch=getchar(),ch>'!');if(flag) x=-x;
}
const int maxn = 100010;
const int lim = 68;
struct Node{
Node *ch[2];
ll lazy;
}mem[maxn*50],*it,*null,*root[maxn];
inline void init(){
it = mem;null = it++;
null->ch[0] = null->ch[1] = null;
null->lazy = 0;root[0] = null;
}
Node* insert(Node *rt,int l,int r,int L,int R){
Node *p = it++;*p = *rt;
if(L <= l && r <= R){
p->lazy ++;
return p;
}
int mid = l+r >> 1;
if(L <= mid) p->ch[0] = insert(p->ch[0],l,mid,L,R);
if(R > mid) p->ch[1] = insert(p->ch[1],mid+1,r,L,R);
return p;
}
ll query(Node *p,int l,int r,int pos){
if(l == r) return p->lazy;
int mid = l+r >> 1;
if(pos <= mid) return query(p->ch[0],l,mid,pos) + p->lazy;
else return query(p->ch[1],mid+1,r,pos) + p->lazy;
}
struct Edge{
int to,next;
}G[maxn<<1];
int head[maxn],cnt;
inline void add(int u,int v){
G[++cnt].to = v;
G[cnt].next = head[u];
head[u] = cnt;
}
ll c[maxn],ind[maxn],dfs_clock,oud[maxn];
#define v G[i].to
void dfs(int u,int f){
ind[u] = ++ dfs_clock;
for(int i = head[u];i;i=G[i].next){
if(v == f) continue;
dfs(v,u);
}oud[u] = dfs_clock;
}
#undef v
struct save{
ll bef,id,val;
save(){}
save(const ll &a,const ll &b,const ll &c){
bef = a;id = b;val = c;
}
}q[maxn];
int q_siz,n;
ull a[maxn],sum[maxn];
inline void rebuild(){
q_siz = 0;
memset(a,0,sizeof a);
for(int i=1;i<=n;++i){
a[ind[i]] += c[i];
}
for(int i=1;i<=n;++i) a[i] += a[i-1];
for(int i=1;i<=n;++i){
sum[i] = sum[i-1] + a[oud[i]] - a[ind[i]-1];
}
}
int main(){
init();
int m;read(n);read(m);
for(int i=1;i<=n;++i) read(c[i]);
int rt = 0;
for(int i=1,u,v;i<=n;++i){
read(u);read(v);
if(u == 0) rt = v;
else add(u,v),add(v,u);
}dfs(rt,0);
for(int i=1;i<=n;++i){
root[i] = insert(root[i-1],1,n,ind[i],oud[i]);
}
rebuild();
ll op,u,v;
while(m--){
read(op);read(u);read(v);
if(op == 1){
q[++q_siz] = save(c[u],u,v);
c[u] = v;
if(q_siz == lim) rebuild();
}else{
ll l = u,r = v;
ull ans = sum[v] - sum[u-1];
for(ll i=1;i<=q_siz;++i){
if(q[i].val - q[i].bef > 0) ans += 1ULL*(q[i].val - q[i].bef)*(query(root[r],1,n,ind[q[i].id]) - query(root[l-1],1,n,ind[q[i].id]));
if(q[i].val - q[i].bef < 0) ans -= 1ULL*(q[i].bef - q[i].val)*(query(root[r],1,n,ind[q[i].id]) - query(root[l-1],1,n,ind[q[i].id]));
}
printf("%llu\n",ans);
}
}
getchar();getchar();
return 0;
}

bzoj 4765: 普通计算姬 主席树+替罪羊树思想的更多相关文章

  1. BZOJ 4765 普通计算姬 dfs序+分块+树状数组+好题!!!

    真是道好题...感到灵魂的升华... 按dfs序建树状数组,拿前缀和去求解散块: 按点的标号分块,分成一个个区间,记录区间子树和 的 总和... 具体地,需要记录每个点u修改后,对每一个块i的贡献,记 ...

  2. BZOJ 4765 普通计算姬 (分块 + BIT)

    4765: 普通计算姬 Time Limit: 30 Sec  Memory Limit: 256 MBSubmit: 1547  Solved: 329[Submit][Status][Discus ...

  3. bzoj 4765 普通计算姬(树状数组 + 分块)

    http://www.lydsy.com/JudgeOnline/problem.php?id=4765 很nice的一道题啊(可能是因为卡了n久终于做出来了 题意就是给你一棵带点权的有根树,sum( ...

  4. bzoj 4765: 普通计算姬

    Description "奋战三星期,造台计算机".小G响应号召,花了三小时造了台普通计算姬.普通计算姬比普通计算机要厉害一些 .普通计算机能计算数列区间和,而普通计算姬能计算树中 ...

  5. bzoj 4765 普通计算姬 dfs序 + 分块

    题目链接 Description "奋战三星期,造台计算机".小G响应号召,花了三小时造了台普通计算姬.普通计算姬比普通计算机要厉害一些.普通计算机能计算数列区间和,而普通计算姬能 ...

  6. BZOJ 4765: 普通计算姬 [分块 树状数组 DFS序]

    传送门 题意: 一棵树,支持单点修改和询问以$[l,r]$为根的子树的权值和的和 只有我这种不会分块的沙茶不会做这道题吗? 说一点总结: 子树和当然上$dfs$序了,询问原序列一段区间所有子树和,对原 ...

  7. BZOJ 4765: 普通计算姬 (分块+树状数组)

    传送门 解题思路 树上的分块题,,对于修改操作,每次修改只会对他父亲到根这条链上的元素有影响:对于查询操作,每次查询[l,r]内所有元素的子树,所以就考虑dfn序,进标记一次,出标记一次,然后子树就是 ...

  8. bzoj 4766: 文艺计算姬 -- 快速乘

    4766: 文艺计算姬 Time Limit: 1 Sec  Memory Limit: 128 MB Description "奋战三星期,造台计算机".小W响应号召,花了三星期 ...

  9. BZOJ 4766: 文艺计算姬

    4766: 文艺计算姬 Time Limit: 1 Sec  Memory Limit: 128 MBSubmit: 456  Solved: 239[Submit][Status][Discuss] ...

随机推荐

  1. 【BZOJ5037】[Jsoi2014]电信网络 最大权闭合图

    [BZOJ5037][Jsoi2014]电信网络 Description JYY创建的电信公司,垄断着整个JSOI王国的电信网络.JYY在JSOI王国里建造了很多的通信基站.目前所有的基站都是使用2G ...

  2. 【BZOJ3640】JC的小苹果 概率DP+高斯消元

    [BZOJ3640]JC的小苹果 Description 让我们继续JC和DZY的故事. “你是我的小丫小苹果,怎么爱你都不嫌多!” “点亮我生命的火,火火火火火!” 话说JC历经艰辛来到了城市B,但 ...

  3. ArcGIS API for js InfoWindow

    说明:有关该示例中怎么引用部署在iis上的离线arcgis api请参考我前面的博文 1.运行效果 2.HTML代码 <!DOCTYPE html> <html> <he ...

  4. springboot工程的结构

    1 springboot的工程结构是什么 就是我们组织springboot工程时遵循的代码的目录结构. 2 spring initializr创建的工程的目录结构 源码目录:src/main/java ...

  5. zookeeper(二): Curator vs zkClient

    目录 zookeeper Curator zkClient 客户端对比 写在前面 1.1. zookeeper应用开发 1.1.1. ZkClient简介 1.1.2. Curator简介 写在最后 ...

  6. They're much closer in spirit to how our brains work than feedforward networks.

    http://neuralnetworksanddeeplearning.com/chap1.html Up to now, we've been discussing neural networks ...

  7. python基础-第六篇-6.4模块混战

    我们之前接触多的编程方式就是函数式编程,而且喜欢就一个文件里写完所有的程序代码,这样做在前期感觉还不错,不过一旦你的程序变复杂,在易读性和排错方面就感觉好吃力,功能界限不明显,那今天我们就来讲讲怎么用 ...

  8. mysql mariadb 乱码

    mysql 创建临时表 CREATE TEMPORARY TABLE tmp_table SELECT COUNT(*) AS num FROM student_info GROUP BY LEFT( ...

  9. ubuntu查看Mysql是否已启动

    sudo netstat -tap | grep mysql 命令行输出: tcp6       0      0 [::]:mysql              [::]:*             ...

  10. 2django 视图与网址进阶

    一.在网页中做加减法 采用/add/?a=11&b=22这样get方法进行 django-admin.py startproject zqxt_views cd zqxt_views pyth ...