Brief Description

给定一棵n个节点的带权树,节点编号为1到n,以root为根,设sum[p]表示以点p为根的这棵子树中所有节点的权

值和。支持下列两种操作:

1 给定两个整数u,v,修改点u的权值为v。

2 给定两个整数l,r,计算sum[l]+sum[l+1]+....+sum[r-1]+sum[r]

Algorithm Design

我们考察暴力算法:

对于查询,我们如果处理出所有的sum[i]就可以处理了。考虑到是树上的子树查询,我们考虑使用dfs序,使用BIT维护即可,这个算法是\(\Theta(log n)\)的。

对于修改,如果我们能够记录每个点是否可以被这个修改的点影响,我们就可以扫一遍就ok了。这样的复杂度是\(\Theta(nlogn)\)的。

我们可以看到修改的复杂度比较大,我们考虑使用分块平衡一下这两个算法,假设我们\(h(n)\)分一块,

对于查询,我们可以把查询区间分为\(\Theta(\frac{n}{h(n)})\)个区间,对于每个区间直接统计,对于两边的区间直接统计复杂度\(\Theta(h(n)log(h(n))\)

对于修改,我们统计每一个区域会被修改的点修改几次,直接统计影响即可。复杂度是\(\Theta(\frac{n}{h(n)})\)

这样,算法总的复杂度就是\(\Theta(h(n)log(h(n))+\frac{n}{h(n)})\),为了方便,我们设\(h(n) = \sqrt n\)。

Code

#include <cmath>
#include <cstdio>
#define ll unsigned long long
const ll maxn = 100010;
const ll maxm = 320;
ll value[maxn], bit[maxn << 1], wtf[maxn];
int f[maxn][maxm];
int t[maxm], l[maxn], r[maxn], head[maxn], b[maxn];
ll n, m, rt, ind = 0, cnt = 0, blockm, block;
struct edge {
ll to, next;
} e[maxn << 2];
ll lowbit(ll x) { return x & -x; }
void change(ll pos, ll x) {
for (ll i = pos; i <= n; i += lowbit(i))
bit[i] += x;
}
ll sum(ll pos) {
ll ans = 0;
for (ll i = pos; i > 0; i -= lowbit(i))
ans += bit[i];
return ans;
}
void insert(ll x, ll y) {
e[++cnt].to = y;
e[cnt].next = head[x];
head[x] = cnt;
e[++cnt].to = x;
e[cnt].next = head[y];
head[y] = cnt;
}
void dfs(ll x, ll fa) {
t[b[x]]++;
for (ll i = 1; i <= blockm; i++)
f[x][i] = t[i];
l[x] = ++ind;
change(l[x], value[x]);
for (ll i = head[x]; i; i = e[i].next) {
if (e[i].to != fa)
dfs(e[i].to, x);
}
r[x] = ind;
t[b[x]]--;
}
int main() {
#ifndef ONLINE_JUDGE
freopen("input", "r", stdin);
#endif
scanf("%llu %llu", &n, &m);
block = (int)sqrt(n);
blockm = (n - 1) / block + 1;
for (ll i = 1; i <= n; i++) {
scanf("%llu", &value[i]);
}
for (ll i = 1; i <= n; i++)
b[i] = (i - 1) / block + 1;
for (ll i = 1; i <= n; i++) {
ll u, v;
scanf("%llu %llu", &u, &v);
if (u == 0)
rt = v;
else
insert(u, v);
}
dfs(rt, 0);
for (ll i = 1; i <= n; i++)
wtf[b[i]] += sum(r[i]) - sum(l[i] - 1);
for (ll i = 1; i <= m; i++) {
ll op, u, v;
scanf("%llu %llu %llu", &op, &u, &v);
if (op == 1) {
change(l[u], v - value[u]);
for (ll j = 1; j <= blockm; j++)
wtf[j] += 1ll * f[u][j] * (v - value[u]);
value[u] = v;
} else {
ll ans = 0;
if (b[u] == b[v])
for (ll j = u; j <= v; j++)
ans += sum(r[j]) - sum(l[j] - 1);
else {
for (ll j = b[u] + 1; j <= b[v] - 1; j++)
ans += wtf[j];
for (ll j = u; b[j] == b[u] && j <= n; j++)
ans += sum(r[j]) - sum(l[j] - 1);
for (ll j = v; b[j] == b[v] && j; j--)
ans += sum(r[j]) - sum(l[j] - 1);
}
printf("%llu\n", ans);
}
}
}

[bzoj4765]普通计算姬——分块的更多相关文章

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

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

  2. BZOJ4765: 普通计算姬

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

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

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

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

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

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

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

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

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

  7. bzoj4765: 普通计算姬 (分块 && BIT)

    最近一直在刷分块啊 似乎感觉分块和BIT是超级棒的搭档啊 这道题首先用dfs预处理一下 得到每一个sum值 此时查询是O(1)的  (前缀和乱搞什么的 但是修改需要O(n) (需要修改该节点所有祖先的 ...

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

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

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

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

随机推荐

  1. RevealTrans图片切换效果

    RevealTrans 更新时间:2013-06-01 17:11:59 | RevealTrans兼容性:IE5.5+ 语法: filter : progid:DXImageTransform.Mi ...

  2. jmeter添加自定义扩展函数之图片base64编码

    打开eclipse,新建maven工程,在pom中引入jmeter核心jar包: <!-- https://mvnrepository.com/artifact/org.apache.jmete ...

  3. 创建vpc网络

    vpc相关功能点: 模块 功能点 描述 备注 VPC 创建vpc网络 创建vpc网络,指定vpc网络名称   修改vpc网络 修改vpc网络名称   删除vpc网络 删除vpc网络   vpc相关命令 ...

  4. 机器学习/逻辑回归(logistic regression)/--附python代码

    个人分类: 机器学习 本文为吴恩达<机器学习>课程的读书笔记,并用python实现. 前一篇讲了线性回归,这一篇讲逻辑回归,有了上一篇的基础,这一篇的内容会显得比较简单. 逻辑回归(log ...

  5. ubuntu 开热点

    原文地址:https://www.cnblogs.com/king-ding/archive/2016/10/09/ubuntuWIFI.html 今天教大家一个简单方法让ubuntu发散wifi热点 ...

  6. Halcon和Opencv区别

    Halcon:机器视觉行业里知名的商业视觉库,非开源的,在国内市场份额处于第一,其提供了1500个多个API算子供开发人员使用,有些编程基础的都可以轻松的入门,其调试也是很方便的,断点单步运行,图像变 ...

  7. sql 先查出已知的数据或者需要的数据再筛选

    sql 先查出已知的数据或者需要的数据再筛选

  8. Java语言常用的运算符和表达式详解

    Java提供了丰富的运算符,如算术运算符.关系运算符.逻辑运算符.位运算符等等.Java的表达式就是用运算符连接起来的符合Java规则的式子.运算符的优先级决定了表达式中运算执行的先后顺序.在编写程序 ...

  9. C#中转义符

    C#转义字符: 一种特殊的字符常量:以反斜线"\"开头,后跟一个或几个字符.具有特定的含义,不同于字符原有的意义,故称“转义”字符.主要用来表示那些用一般字符不便于表示的控制代码. ...

  10. The XOR Largest Pair

    刷刷书上的例题 在给定的N个整数A1,A2……An中选出两个进行XOR运算,得到的结果最大是多少?N<=105,0<=Ai<231 SOlution: 我们思考到对于两个数相异或,是 ...