http://www.lydsy.com/JudgeOnline/problem.php?id=4765

很nice的一道题啊(可能是因为卡了n久终于做出来了

题意就是给你一棵带点权的有根树,sum(i)表示以i为根的这颗子树中所有节点的权值和。有两种操作,一种是修改某个点的权值,另一种是给出l,r,求sum(l)+sum(l+1)...+sum(r)。

首先考虑一个简单的问题,如果单求其中一个sum(i),我们可以怎样做。

很明显我们画个图,我们可以看到每个点打上dfs序之后,每个sum就变成了一个区间,那就是单点更新,区间求和了,树状数组可以很好的解决掉。

那现在我们要解决原问题了,因为没学过树套树的玩意,所以我觉得有了上面那个东西之后,是不是可以用树套树来搞呢,查了好久,没发现什么资料,也不知道能不能搞,反正我是不会了= =,然后就尝试莫队的东西,发现更新我也不会啊,于是就傻逼了。

对大佬说了很多傻逼想法然后全被自己否决了,比如说分块一下sum啊(然而这个我自己否决了的就是最后A掉的做法),或者是,之后,突然感觉有几个不可行的合起来好像复杂度很对啊。再瞎画一下,得出了个想法。

就是上面说的分块一下sum,分块,看起来复杂度很支持啊,但是我们要考虑到,修改一个点,会使得很多个sum发生改变,我一开始否决就是把这里的复杂度算错了。那其实这里我们可以用一个贡献一样的数组来求,gx[i][j]表示修改j对第i块会产生多大影响(就是说j在第i块出现过几次),这个只需要nsqrtn就可以求出来。一开始我以为要n^2???我别是傻子吧!

但是除了整块之外的,还有其它那些,这里我们就暴力用树状数组同时维护起来,于是整体复杂度就变成了优美的nsqrtnlogn!

回忆一下貌似自己想到简化的那个问题之后,就不应该想其他东西啊,来个暴力分块就好了,不过还好总算是A掉了,另外此题答案爆longlong。。

/**************************************************************
Problem: 4765
User: caobao
Language: C++
Result: Accepted
Time:13716 ms
Memory:136396 kb
****************************************************************/ #include <cstdio>
#include <cstring>
#include <cmath>
#include <queue>
#include <vector>
#include <time.h>
#include <string>
#include <stack>
#include <set>
#include <map>
#include <iostream>
#include <bitset>
#include <algorithm>
using namespace std;
#define MP make_pair
#define PB push_back
#define ms(a,b) memset((a),(b),sizeof(a))
typedef long long LL;
typedef unsigned long long uLL;
typedef pair<int, int> Pii;
typedef vector<int> Vi;
typedef vector<Pii> Vii;
const int inf = 0x3f3f3f3f;
const LL INF = (1uLL << ) - ;
const LL mod = ;
const int N = + ;
const double Pi = acos(-1.0);
const int maxn = ;
int head[maxn];
int w[maxn];
int bl[maxn];
uLL Vsum[];
int tol;
int g;
int n, m;
int root;
int sz;
struct Edge {
int to, nx;
} edge[maxn << ];
void init() {
memset(head, -, sizeof head);
}
inline void add_edge(int u, int v) {
edge[tol].to = v;
edge[tol].nx = head[u];
head[u] = tol++;
}
inline int lowbit(int x) {
return x & (-x);
}
inline void OT(uLL x) {
if(x > ) {
OT(x / );
}
putchar(x % + '');
}
uLL tree[maxn];
struct Ds {
int l, r;
} S[maxn];
inline uLL sum(int pos) {
uLL res = ;
for(; pos; pos -= lowbit(pos))res += tree[pos];
return res;
}
inline void fix(int pos, uLL x) {
x = x - sum(pos) + sum(pos - );
for(; pos < maxn; pos += lowbit(pos))tree[pos] += x;
}
void dfs(int u, int fa) {
S[u].l = ++g;
for(int i = head[u]; ~i; i = edge[i].nx) {
int v = edge[i].to;
if(v == fa)continue;
dfs(v, u);
}
S[u].r = g;
}
int line[maxn + ];
int gx[][maxn];
inline int id(int x) {
return S[x].l;
}
inline void udate(int pos, uLL x) {
fix(id(pos), x);
uLL add = x - w[pos];
w[pos] = x;
for(int i = ; i <= bl[n]; i++) {
Vsum[i] += gx[i][id(pos)] * add;
}
}
inline uLL ask(int l, int r) {
uLL res = ;
if(bl[l] == bl[r]) {
for(int i = l; i <= r; i++) {
res += sum(S[i].r) - sum(S[i].l - );
}
return res;
}
for(int i = l; bl[i] == bl[l]; i++) {
res += sum(S[i].r) - sum(S[i].l - );
}
for(int i = bl[l] + ; i < bl[r]; i++) {
res += Vsum[i];
}
for(int i = (bl[r] - ) * sz + ; i <= r; i++) {
res += sum(S[i].r) - sum(S[i].l - );
}
return res;
}
int main() {
#ifdef local
freopen("in", "r", stdin);
// freopen("w","w",stdout);
#endif
//ios::sync_with_stdio(false);
// cin.tie(0);
init();
scanf("%d%d", &n, &m);
sz = sqrt(n);
for(int i = ; i <= n; ++i)scanf("%d", &w[i]);
for(int i = ; i <= n; ++i) {
bl[i] = (i - ) / sz + ;
}
for(int i = ; i < n; ++i) {
int a, b;
scanf("%d%d", &a, &b);
if(a)add_edge(a, b), add_edge(b, a);
else root = b;
}
dfs(root, -);
for(int i = ; i <= n; ++i)fix(S[i].l, w[i]);
for(int i = ; i <= bl[n]; ++i) {
memset(line, , sizeof line);
for(int j = (i - ) * sz + ; j <= min(i * sz , n); j++) {
line[S[j].l]++, line[S[j].r + ]--;
Vsum[i] += sum(S[j].r) - sum(S[j].l - );
}
int tmp = ;
for(int j = ; j < maxn; j++) {
tmp += line[j];
gx[i][j] = tmp;
}
}
for(int i = ; i < m; i++) {
int d, l, r;
scanf("%d%d%d", &d, &l, &r);
if(d == ) {
udate(l, r);
} else OT(ask(l, r)), putchar('\n');
}
}

-------另外,发现数据的修改权值都是往大修改的,因为里面修改的时候,我传得都是uLL,但ac,懒得改了,看的人注意就可以了。-------

bzoj 4765 普通计算姬(树状数组 + 分块)的更多相关文章

  1. BZOJ_2141_排队_树状数组+分块

    BZOJ2141_排队_树状数组+分块 Description 排排坐,吃果果,生果甜嗦嗦,大家笑呵呵.你一个,我一个,大的分给你,小的留给我,吃完果果唱支歌,大家 乐和和.红星幼儿园的小朋友们排起了 ...

  2. gym 100589A queries on the Tree 树状数组 + 分块

    题目传送门 题目大意: 给定一颗根节点为1的树,有两种操作,第一种操作是将与根节点距离为L的节点权值全部加上val,第二个操作是查询以x为根节点的子树的权重. 思路: 思考后发现,以dfs序建立树状数 ...

  3. 【bzoj4889】[Tjoi2017]不勤劳的图书管理员 树状数组+分块+二分

    题目描述(转自洛谷) 加里敦大学有个帝国图书馆,小豆是图书馆阅览室的一个书籍管理员.他的任务是把书排成有序的,所以无序的书让他产生厌烦,两本乱序的书会让小豆产生这两本书页数的和的厌烦度.现在有n本被打 ...

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

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

  5. bzoj 2527 Meteors - 整体二分 - 树状数组

    Description Byteotian Interstellar Union (BIU) has recently discovered a new planet in a nearby gala ...

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

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

  7. BZOJ 1901 Zju2112 Dynamic Rankings ——树状数组套主席树

    [题目分析] BZOJ这个题目抄的挺霸气. 主席树是第一时间想到的,但是修改又很麻烦. 看了别人的题解,原来还是可以用均摊的思想,用树状数组套主席树. 学到了新的姿势,2333o(* ̄▽ ̄*)ブ [代 ...

  8. BZOJ 3224 普通平衡树(树状数组)

    题目链接:http://61.187.179.132/JudgeOnline/problem.php?id=3224 题意:维护以下操作:(1)插入x:(2)删除x(若有多个相同的数,只删除一个)(3 ...

  9. [BZOJ 1901] Dynamic Rankings 【树状数组套线段树 || 线段树套线段树】

    题目链接:BZOJ - 1901 题目分析 树状数组套线段树或线段树套线段树都可以解决这道题. 第一层是区间,第二层是权值. 空间复杂度和时间复杂度均为 O(n log^2 n). 线段树比树状数组麻 ...

随机推荐

  1. coolpad 5879logcat不能输入日志解决办法

    有几天没完手机了,玩一下,发现不能打印日志了,记得最开始的时候 会弹出一个选项选择是否输出日志,在网上找了好的方法. 1.重启adb,点击DBMS 进行刷新. 2重启eclipse. 3.重装驱动. ...

  2. Reflux中文教程——概览

    翻译自github上的reflux项目,链接:https://github.com/reflux/refluxjs 〇.安装及引入 安装: npm install reflux 引入: var Ref ...

  3. [Paxos] Paxos Made Simple 读后感

    Paxos 由著名图灵奖获得者Leslie Lamport提出,该算法是分布式一致性算法中的奠基之作,今天初读此文仅将相关学习心得予以记录. 1.Paxos 是什么?主要用来解决什么问题? Paxos ...

  4. 利用sub lr,lr,#4:程序是如何进行返回的?

    1: ARM采用的是3级流水线 ARM的流水线结构为:   取指 -----> 译码 ------> 执行 ARM代码:                  PC           PC- ...

  5. 轻量级操作系统FreeRTOS的内存管理机制(一)

    本文由嵌入式企鹅圈原创团队成员朱衡德(Hunter_Zhu)供稿. 近几年来,FreeRTOS在嵌入式操作系统排行榜中一直位居前列,作为开源的嵌入式操作系统之一,它支持许多不同架构的处理器以及多种编译 ...

  6. 第八章 Hibernate数据操作技巧

    第八章   Hibernate数据操作技巧8.1 分组统计数据    语法:[select]... from ...[where] [group by...[having...]] [order by ...

  7. HTML表单基本格式与代码

    咱们先来看下今天咱们需要学习的内容,理解起来很简单,像我这种英语不好的只是需要背几个单词 在HTML中创建表单需要用到的最基本的代码和格式 <form method="post/get ...

  8. js中元素(图片)切换和隐藏显示问题

    这个知识点其实也简单,(当然是在理清思路的情况下),在没预习的情况下听的还真是艰难,上课以来唯一的一次懵逼了一天,感觉乱乱的,全是新属性,所以今晚的我破天荒的熬夜敲代码,一定要弄懂! 现在就来梳理下头 ...

  9. CSAcademy Beta Round #5 Long Journey

    题目链接:https://csacademy.com/contest/arhiva/#task/long_journey/ 大意是有一张无向不带权的图,两个人同时从s点出发,分别前往a点和b点,且每个 ...

  10. leetcode水题(一)

    Two Sum 1 public int[] twoSum(int[] numbers,int target){ Map<Integer,Integer> map = new HashMa ...