Gardener Bo (树剖 + 问题分解)
一开始没看懂计算答案的第四部和update2,很迷。然后一直推敲之后才发下我计算的时候漏掉一个关键点。没有把加值的影响放到父节点上。
#include<bits/stdc++.h>
#define M ((l + r) / 2)
#define ls (rt << 1)
#define rs ((rt << 1) | 1)
#define UI unsigned int
using namespace std; const int maxn = 6e5 + ;
vector<int>Gra[maxn];
UI f[maxn], c[maxn], sz[maxn], w[maxn], sum[maxn << ], add[maxn], cntSon[maxn], cntSeson[maxn];
int fa[maxn], N, top[maxn], hSon[maxn];
int edn[maxn], stn[maxn]; /**
stn dfs序的起点, edn dfs序的终点,hSon 重树,add 加重, cntSon 直接子节点数量
sz 树的总大小, cntSeson 2层子节点的总数量, c 子树重复计算个数,
f 未进行操作的子树总值 top 重链的父节点。
**/ ///主要是计算初始值
void dfs1(int u) {
cntSeson[u] = cntSon[u] = hSon[u] = c[u] =;
sz[u] = ;
for(auto v: Gra[u]) {
if(v == fa[u]) continue;
dfs1(v);
sz[u] += sz[v];
cntSon[u] ++;
cntSeson[u] += cntSon[v] + ;
c[u] += sz[v] * cntSon[v];
if(sz[hSon[u]] < sz[v]) hSon[u] = v;
}
f[u] = w[u] * (sz[u] + );
for(auto v: Gra[u]) {
if(v == fa[u]) continue;
f[u] += (sz[u] - sz[v]) * w[v];
w[u] += w[v];
}
} ///dfs序,以及最重要的重树父节点
void dfs2(int u, int t) {
top[u] = t;
stn[u] = ++N;
if(hSon[u]) dfs2(hSon[u], t);
for(auto v: Gra[u]) {
if(v == fa[u] || hSon[u] == v) continue;
dfs2(v, v);
}
edn[u] = N;
} ///查询线段树上的价值
int query(int L, int R, int l, int r, int rt) {
if(L <= l && r <= R) return sum[rt];
int ret = ;
if(M >= L) ret += query(L, R, l, M, ls);
if(M < R) ret += query(L, R, M + , r, rs);
return ret;
} ///作用之后对父节点的影响
void update1(int x, int val) {
while(top[x] != top[]) {
int v = top[x], u = fa[v];
f[u] += (sz[u] - sz[v]) * val;
x = u;
}
} ///存加的价值的影响范围
void update2(int p, UI val, int l, int r, int rt) {
sum[rt] += val;
if(l == r) return ;
if(M >= p) update2(p, val, l, M, ls);
else update2(p, val, M + , r, rs);
} int main() {
freopen("data.txt", "r", stdin);
int n, m, st, ed;
while(~scanf("%d%d", &n, &m)) {
memset(add, , sizeof(add));
memset(sum, , sizeof(sum));
N = ;
for(int i = ; i <= n; i ++)
Gra[i].clear();
for(int i = ; i <= n; i ++) {
scanf("%d", &ed);
fa[i] = ed;
Gra[ed].push_back(i);
Gra[i].push_back(ed);
}
for(int i = ; i <= n; i ++)
scanf("%u", &w[i]);
dfs1();
dfs2(, );
while(m -- ) {
int op, u;
UI x;
scanf("%d", &op);
if(op == ) {
scanf("%d%u", &u, &x);
add[u] += x;
///沿着重树往上预计算重树点价值的作用,之后就不用重复计算这一部分了,这样重树这里就可以成为一个撬棍
update1(u, x * ( + cntSeson[u]));
///将每个点的填加的价值放进去线段树
update2(stn[u], x * (cntSeson[u]+ ), , n, );
} else {
scanf("%d", &u);
UI ans = f[u];
///加值时作用在当前节点上的情况
ans += add[u] * ( + sz[u] * cntSeson[u] - c[u]);
if(fa[u])///加值时作用在子节点上的情况
ans += add[fa[u]] * ( + sz[u] * cntSon[u]);
if(fa[fa[u]])///加值时作用在孙子节点上的情况
ans += add[fa[fa[u]]] * ( + sz[u]);
if(hSon[u])///计算除重树内加值的价值,因为已经在update1计算过了到f中了
ans += (sz[u] - sz[hSon[u]]) * query(stn[hSon[u]], edn[hSon[u]], , n, );
printf("%u\n",ans);
}
}
}
return ;
}
Gardener Bo (树剖 + 问题分解)的更多相关文章
- 【61测试】【dp】【二分】【前缀和】【树剖】
不要问我为什么昨天考的今天才贴解题报告.. 第一题: 给定3个字符串,求它们的最长公共子序列. 解: 考试时知道肯定是LCS的二维再加一维,用三维,可天堂有路你不走,地狱无门你偏来...灵机一动想出来 ...
- hdu_5221_Occupation(树剖)
题目连接:http://acm.hdu.edu.cn/showproblem.php?pid=5221 题意:给你一棵树,每个节点有一定的值,有三种操作: 1 x y 表示占领树上x-y的所有节点,2 ...
- 【树链剖分】洛谷P3384树剖模板
题目描述 如题,已知一棵包含N个结点的树(连通且无环),每个节点上包含一个数值,需要支持以下操作: 操作1: 格式: 1 x y z 表示将树从x到y结点最短路径上所有节点的值都加上z 操作2: 格式 ...
- 【BZOJ 3626】 [LNOI2014]LCA【在线+主席树+树剖】
题目链接: TP 题解: 可能是我比较纱布,看不懂题解,只好自己想了…… 先附一个离线版本题解[Ivan] 我们考虑对于询问区间是可以差分的,然而这并没有什么卵用,然后考虑怎么统计答案. 首先LC ...
- BZOJ_2238_Mst_树剖+线段树
BZOJ_2238_Mst_树剖+线段树 Description 给出一个N个点M条边的无向带权图,以及Q个询问,每次询问在图中删掉一条边后图的最小生成树.(各询问间独立,每次询问不对之后的询问产生影 ...
- BZOJ_3626_[LNOI2014]LCA_离线+树剖
BZOJ_3626_[LNOI2014]LCA_离线+树剖 题意: 给出一个n个节点的有根树(编号为0到n-1,根节点为0).一个点的深度定义为这个节点到根的距离+1. 设dep[i]表示点i的深度, ...
- BZOJ_2588_Spoj 10628. Count on a tree_树剖+主席树
BZOJ_2588_Spoj 10628. Count on a tree_树剖+主席树 题意: 给定一棵N个节点的树,每个点有一个权值,对于M个询问(u,v,k),你需要回答u xor lastan ...
- BZOJ_4551_[Tjoi2016&Heoi2016]树_树剖+线段树
BZOJ_4551_[Tjoi2016&Heoi2016]树_树剖+线段树 Description 在2016年,佳媛姐姐刚刚学习了树,非常开心.现在他想解决这样一个问题:给定一颗有根树(根为 ...
- BZOJ_2157_旅游_树剖+线段树
BZOJ_2157_旅游_树剖+线段树 Description Ray 乐忠于旅游,这次他来到了T 城.T 城是一个水上城市,一共有 N 个景点,有些景点之间会用一座桥连接.为了方便游客到达每个景点但 ...
随机推荐
- 洛谷P1955 程序自动分析 [NOI2015] 并查集
正解:并查集+离散化 解题报告: 传送门! 其实题目还挺水的,,,但我太傻逼了直接想挂了,,,所以感觉还是有个小坑点所以还是写个题解记录下我的傻逼QAQ 首先这题一看,就长得很像NOIp关押罪犯?然后 ...
- hotplug 热拔插机制框架
框架入口源文件: mdev.c (可根据入口源文件,再按着框架到内核走一遍) 内核版本:linux_2.6.22.6 硬件平台:JZ2440 以下是驱动框架:
- 关于安装SVN Service 出错 Service 'VisualSVN Server' failed to start. Please check VisualSVN Server log in Event Viewer for more details
关于安装SVN Service 出错 Service 'VisualSVN Server' failed to start. Please check VisualSVN Server log in ...
- RSA 理论
一.同余 给定一个正整数m,如果两个整数a和b满足(a-b)能够被m整除,即(a-b)/m得到一个整数,那么就称整数a与b对模m同余,记作a≡b(mod m) 二.欧拉定理 任意给定正整数n,请问在小 ...
- Python3学习之路~6.5 私有属性和私有方法
属性分为静态属性和动态属性,静态属性就是变量,动态属性就是方法.但是一般我们说的属性就是变量,方法就是方法.私有属性/方法就是外面访问不了,只有自己能够访问的属性/方法.如何将属性和方法分别变成私有属 ...
- Jmeter测试实践:文件上传接口
1.打开jmeter4.0,新建测试计划,添加线程组.根据实际情况配置线程属性. 2.添加HTTP请求. Basic部分修改如下: Advanced部分我做任何修改,完全保持默认.但是有人说Imple ...
- 并发编程---Process对象的其他属性或方法
Process对象的其他属性或方法 #join方法 from multiprocessing import Process import time,os ''' 需求:让主进程等着子进程运行完毕,才能 ...
- 关于获取路径path
String webPath = request.getServletPath(); log.info(webPath); 输出: /zjdlbb/zjdlbb/zjdlbb/test.ht log. ...
- js计算常见操作
如何实现数字相加 var a = 1, var b = 2, var c = a + b 这样c得出来的解果是12, 使用Number()函数可以解决这个问题,如下 var c = Number(a) ...
- rpm: /root/anaconda3/lib/liblzma.so.5: version `XZ_5.1.2alpha' not found (required by /lib64/librpmio.so.3)
报如上的错误,发现rpm相关的命令都不能够用. 1.搜到这篇文章,https://stackoverflow.com/questions/47633870/rpm-lib64-liblzma-so-5 ...