最近一直在刷分块啊

似乎感觉分块和BIT是超级棒的搭档啊

这道题首先用dfs预处理一下

得到每一个sum值

此时查询是O(1)的  (前缀和乱搞什么的

但是修改需要O(n) (需要修改该节点所有祖先的sum

复杂度就爆了呀

此时考虑分块优化

似乎弹飞绵羊也是这样思考得出分块做法的

首先分成 √n 块

sum[i]记录第i块的sum和

中间的块直接用sum数组处理  两边用树状数组暴力求

这样查询就是O(√n)的 (其实有一些常数的...  就当是 √n 好了)

修改的话在dfs时用f[i][j]表示第j个点对于第i块的贡献 (需要算几次什么的

然后直接修改块就好了  复杂度也是O (√n) 的

下面是代码

 #include <cstdio>
#include <cmath>
#include <algorithm>
#include <vector>
using namespace std;
#define isdigit(x) (x >= '0' && x <= '9')
#define lowbit(x) (x & (-x))
typedef unsigned long long ll;
const int N = 1e5 + ;
const int M = ; int n, root, cnt, sz, tot;
int d[N], b[N], f[M][N], ct[N], L[N], p[N];
ll s[N], c[N];
vector < int > E[N]; inline void read(int &ans) {
ans = ;
register int res = ;
static char buf = getchar();
for (; !isdigit(buf); buf = getchar())
if (buf == '-') res = -;
for (; isdigit(buf); buf = getchar())
ans = ans * + buf - '';
ans *= res;
} inline void addEdge(int u ,int v) {
E[u].push_back(v);
E[v].push_back(u);
} inline void add(int x, ll v) {
while (x <= n) {
c[x] += v;
x += lowbit(x);
}
} inline ll query(int x) {
ll ans = ;
while (x > ) {
ans += c[x];
x -= lowbit(x);
}
return ans;
} ll dfs(int x, int fa) {
ll sum = d[x]; p[x] = ++tot;
ct[b[x]]++; add(tot, d[x]);
for (int i = ; i <= cnt; i++) f[i][x] += ct[i];
for (int i = ; i < E[x].size(); i++) {
int u = E[x][i];
if (u == fa) continue;
sum += dfs(u, x);
}
ct[b[x]]--; L[x] = tot;
s[b[x]] += sum;
return sum;
} inline void modify(int u, int v) {
add(p[u], v - d[u]);
for (int i = ; i <= cnt; i++)
s[i] += (v - d[u]) * 1ll * f[i][u];
d[u] = v;
} inline ll query(int l ,int r) {
ll ans = ;
if (b[l] == b[r]) {
for (int i = l; i <= r; i++)
ans += query(L[i]) - query(p[i] - );
return ans;
}
for (int i = b[l] + ; i < b[r]; i++)
ans += s[i];
for (int i = l; i <= b[l] * sz; i++)
ans += query(L[i]) - query(p[i] - );
for (int i = (b[r] - ) * sz + ; i <= r; i++)
ans += query(L[i]) - query(p[i] - );
return ans;
} int main() {
int m;
read(n); read(m);
sz = sqrt(n);
for (int i = ; i <= n; i++) {
read(d[i]);
b[i] = (i - ) / sz + ;
}
cnt = b[n];
for (int i = ; i <= n; i++) {
int u, v;
read(u); read(v);
if (!u) root = v;
else addEdge(u, v);
}
dfs(root, );
while (m--) {
int op, u, v;
read(op); read(u); read(v);
if (op == )
modify(u, v);
else
printf("%llu\n", query(u, v));
}
return ;
}

bzoj4765: 普通计算姬 (分块 && BIT)的更多相关文章

  1. [BZOJ4765]普通计算姬(分块+树状数组)

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

  2. [bzoj4765]普通计算姬——分块

    Brief Description 给定一棵n个节点的带权树,节点编号为1到n,以root为根,设sum[p]表示以点p为根的这棵子树中所有节点的权 值和.支持下列两种操作: 1 给定两个整数u,v, ...

  3. BZOJ4765: 普通计算姬

    BZOJ4765: 普通计算姬 题目描述 传送门 题目分析 求的和非常奇怪,不具有连续性,所有上树的数据结构全死了. 考虑分块,思考对于一段连续的询问区间可以直接询问整块,零散块可以在树上dfs序暴力 ...

  4. 2018.06.30 BZOJ4765: 普通计算姬(dfs序+分块+树状数组)

    4765: 普通计算姬 Time Limit: 30 Sec Memory Limit: 256 MB Description "奋战三星期,造台计算机".小G响应号召,花了三小时 ...

  5. [bzoj4765]普通计算姬(分块+树状数组+DFS序)

    题意 给定一棵n个节点的带权树,节点编号为1到n,以root为根,设sum[p]表示以点p为根的这棵子树中所有节点的权值和.计算姬支持下列两种操作: 1 给定两个整数u,v,修改点u的权值为v. 2 ...

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

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

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

    对节点按编号分块.设f[i][j]为修改j号点对第i块的影响,计算f[i][]时dfs一遍即可.记录每一整块的sum.修改时对每一块直接更新sum,同时用dfs序上的树状数组维护子树和.查询时累加整块 ...

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

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

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

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

随机推荐

  1. 二分类模型之logistic

    liner classifiers 逻辑回归用在2分类问题上居多.它是一个非线性的回归模型,其最大的好处恰恰是可以解决二元类问题,目前在金融行业,基本都是使用Logistic回归来预判一个用户是否为好 ...

  2. vue登录管理

    pc端页面登录逻辑,用户未登录状态下可以访问所有页面,但是请求数据是有限的,只有在登录状态下才能访问全部数据,同时未登录状态下,可视区有遮挡元素提示登录. 主要使用的技术vue.vuex.vue-ro ...

  3. Zjnu Stadium HDU - 3047 带权并查集板子题

    #include<iostream> #include<cstring> #include<cstdio> using namespace std; +; int ...

  4. Docker学习参考网站

    Docker——从入门到实践 https://www.yuque.com/grasilife/docker 2.阿里源网站:https://cr.console.aliyun.com 3.DaoClo ...

  5. js数据类型1

    1. 分类(2大类) 基本(值)类型--5种 Number: 任意数值 String: 任意文本 Boolean: true/false undefined: undefined null: null ...

  6. 【Unity|C#】基础篇(21)——常用类

  7. C语言 sizeof()用法介绍

    本文 转自https://www.cnblogs.com/huolong-blog/p/7587711.html   1.      定义 sizeof是一个操作符(operator). 其作用是返回 ...

  8. 【python&pycharm的安装使用】

    一.Python3.7安装 1. 运行python3.7.exe 2. 检查是否安装成功:命令窗口输入python -V 二.Pycharm安装 1. 运行pycharm.exe(社区版) 2. 配置 ...

  9. ubantu安装apache

    1.命令安装: sudo apt install apache2 2.检查是否启动了Apache服务 systemctl status apache2 3.开启.关闭和重启服务器 /etc/init. ...

  10. Bootstrap框架中radio设置值

    Bootstrap中的radio设置值不能像我们平常给普通radio赋值那样,因为无效. 我们用Bootstrap框架里的radio组件,代码: <div class="radio-l ...