Solution -「BZOJ #3786」星系探索
\(\mathcal{Description}\)
Link.
给定一棵含 \(n\) 个点的有根树,点有点权,支持 \(q\) 次操作:
- 询问 \(u\) 到根的点权和;
- 修改 \(u\) 的父亲,保证得到的图仍是树;
- 将 \(u\) 子树内的点权增加一常数。
\(n\le10^5\),\(q\le3\times10^5\)。
\(\mathcal{Solution}\)
ETT (Euler Tour Tree),是一种能快速处理子树移动的动态树。本质上,它将树保存作欧拉序,由于子树移动体现在欧拉序上是区间移动,那么就能使用平衡树维护序列。路径查询亦对应了一段区间内仅出现一次的结点贡献和,记每个结点第一次出现为正贡献,后一次出现为负贡献就变成了区间查询,亦能用平衡树维护。本题即 ETT 板题。
复杂度自然是 \(\mathcal O((n+q)\log n)\) 的。
\(\mathcal{Code}\)
/*~Rainybunny~*/
#include <cstdio>
#include <cassert>
#include <cstdlib>
#define rep( i, l, r ) for ( int i = l, rep##i = r; i <= rep##i; ++i )
#define per( i, r, l ) for ( int i = r, per##i = l; i >= per##i; --i )
typedef long long LL;
const int MAXN = 1e5;
int n, ecnt, head[MAXN + 5], w[MAXN + 5], root, st[MAXN + 5], ed[MAXN + 5];
struct Edge { int to, nxt; } graph[MAXN + 5];
inline void link( const int s, const int t ) {
graph[++ecnt] = { t, head[s] }, head[s] = ecnt;
}
struct Treap {
static const int MAXND = MAXN << 1;
int node, ch[MAXND + 5][2], fa[MAXND + 5], siz[MAXND + 5], aux[MAXND + 5],
sign[MAXND + 5], ssiz[MAXND + 5];
LL val[MAXND + 5], sum[MAXND + 5], tag[MAXND + 5];
Treap() { srand( 20120712 ); }
inline int crtnd( const int v, const int s ) {
int u = ++node;
aux[u] = rand(), siz[u] = 1;
sign[u] = ssiz[u] = s;
val[u] = sum[u] = v * s;
return u;
}
inline void pushad( const int u, const LL x ) {
val[u] += sign[u] * x, tag[u] += x, sum[u] += x * ssiz[u];
}
inline void pushup( const int u ) {
siz[u] = siz[ch[u][0]] + siz[ch[u][1]] + 1;
ssiz[u] = ssiz[ch[u][0]] + ssiz[ch[u][1]] + sign[u];
sum[u] = sum[ch[u][0]] + sum[ch[u][1]] + val[u];
}
inline void pushdn( const int u ) {
if ( tag[u] ) {
if ( ch[u][0] ) pushad( ch[u][0], tag[u] );
if ( ch[u][1] ) pushad( ch[u][1], tag[u] );
tag[u] = 0;
}
}
inline int merge( const int u, const int v ) {
if ( !u || !v ) return u | v;
pushdn( u ), pushdn( v );
if ( aux[u] < aux[v] ) {
ch[u][1] = merge( ch[u][1], v );
if ( ch[u][1] ) fa[ch[u][1]] = u;
return pushup( u ), u;
} else {
ch[v][0] = merge( u, ch[v][0] );
if ( ch[v][0] ) fa[ch[v][0]] = v;
return pushup( v ), v;
}
}
inline void rsplit( const int u, const int k, int& x, int& y ) {
if ( !u ) return void( x = y = 0 );
fa[u] = 0, pushdn( u );
if ( siz[ch[u][0]] >= k ) {
y = u, rsplit( ch[u][0], k, x, ch[y][0] );
if ( ch[y][0] ) fa[ch[y][0]] = y;
} else {
x = u, rsplit( ch[u][1], k - siz[ch[u][0]] - 1, ch[x][1], y );
if ( ch[x][1] ) fa[ch[x][1]] = x;
}
pushup( u );
}
inline int rank( int u ) {
int ret = siz[ch[u][0]] + 1;
while ( fa[u] ) {
if ( ch[fa[u]][1] == u ) ret += siz[ch[fa[u]][0]] + 1;
u = fa[u];
}
return ret;
}
} trp;
inline void init( const int u ) {
root = trp.merge( root, st[u] = trp.crtnd( w[u], 1 ) );
for ( int i = head[u]; i; i = graph[i].nxt ) init( graph[i].to );
root = trp.merge( root, ed[u] = trp.crtnd( w[u], -1 ) );
}
inline LL query( const int u ) {
int l = trp.rank( st[1] ), r = trp.rank( st[u] ), x, y, z;
assert( l == 1 );
trp.rsplit( root, l - 1, x, y ), trp.rsplit( y, r - l + 1, y, z );
LL ret = trp.sum[y];
root = trp.merge( x, trp.merge( y, z ) );
return ret;
}
inline void change( const int u, const int p ) {
int l = trp.rank( st[u] ), r = trp.rank( ed[u] ), tar = trp.rank( st[p] ),
x, y, z, m;
if ( tar < l ) {
trp.rsplit( root, tar, x, y );
trp.rsplit( y, l - tar - 1, y, z );
trp.rsplit( z, r - l + 1, z, m );
} else {
trp.rsplit( root, l - 1, x, y );
trp.rsplit( y, r - l + 1, y, z );
trp.rsplit( z, tar - r, z, m );
}
root = trp.merge( trp.merge( x, z ), trp.merge( y, m ) );
}
inline void update( const int u, const int p ) {
int l = trp.rank( st[u] ), r = trp.rank( ed[u] ), x, y, z;
trp.rsplit( root, l - 1, x, y ), trp.rsplit( y, r - l + 1, y, z );
trp.pushad( y, p ), root = trp.merge( x, trp.merge( y, z ) );
}
int main() {
scanf( "%d", &n );
for ( int i = 2, t; i <= n; ++i ) scanf( "%d", &t ), link( t, i );
rep ( i, 1, n ) scanf( "%d", &w[i] );
init( 1 );
char op[5]; int q, a, b;
for ( scanf( "%d", &q ); q--; ) {
scanf( "%s %d", op, &a );
if ( op[0] == 'Q' ) printf( "%lld\n", query( a ) );
else if ( op[0] == 'C' ) scanf( "%d", &b ), change( a, b );
else scanf( "%d", &b ), update( a, b );
}
return 0;
}
Solution -「BZOJ #3786」星系探索的更多相关文章
- Solution -「BZOJ 3812」主旋律
\(\mathcal{Description}\) Link. 给定含 \(n\) 个点 \(m\) 条边的简单有向图 \(G=(V,E)\),求 \(H=(V,E'\subseteq E)\ ...
- 【bzoj 3786】星系探索
Description 物理学家小C的研究正遇到某个瓶颈. 他正在研究的是一个星系,这个星系中有n个星球,其中有一个主星球(方便起见我们默认其为1号星球),其余的所有星球均有且仅有一个依赖星球.主星球 ...
- Solution -「BZOJ 4316」小C的独立集
\(\mathcal{Description}\) Link. 求包含 \(n\) 个结点 \(m\) 条边的仙人掌的最大独立集. \(n\le5\times10^4\),\(m\le6\ ...
- Solution -「BZOJ 3331」压力
\(\mathcal{Description}\) Link. 给定一个 \(n\) 个点 \(m\) 条边的连通无向图,并给出 \(q\) 个点对 \((u,v)\),令 \(u\) 到 \ ...
- 「BZOJ 4228」Tibbar的后花园
「BZOJ 4228」Tibbar的后花园 Please contact lydsy2012@163.com! 警告 解题思路 可以证明最终的图中所有点的度数都 \(< 3\) ,且不存在环长是 ...
- 「BZOJ 3645」小朋友与二叉树
「BZOJ 3645」小朋友与二叉树 解题思路 令 \(G(x)\) 为关于可选大小集合的生成函数,即 \[ G(x)=\sum[i\in c ] x^i \] 令 \(F(x)\) 第 \(n\) ...
- 「BZOJ 4502」串
「BZOJ 4502」串 题目描述 兔子们在玩字符串的游戏.首先,它们拿出了一个字符串集合 \(S\),然后它们定义一个字符串为"好"的,当且仅当它可以被分成非空的两段,其中每一段 ...
- 「BZOJ 4289」 PA2012 Tax
「BZOJ 4289」 PA2012 Tax 题目描述 给出一个 \(N\) 个点 \(M\) 条边的无向图,经过一个点的代价是进入和离开这个点的两条边的边权的较大值,求从起点 \(1\) 到点 \( ...
- 「BZOJ 2534」 L - gap字符串
「BZOJ 2534」 L - gap字符串 题目描述 有一种形如 \(uv u\) 形式的字符串,其中 \(u\) 是非空字符串,且 \(v\) 的长度正好为 \(L\), 那么称这个字符串为 \( ...
随机推荐
- 使用swagger生成API文档
有时候一份清晰明了的接口文档能够极大地提高前后端双方的沟通效率和开发效率.本文将介绍如何使用swagger生成接口文档. swagger介绍 Swagger本质上是一种用于描述使用JSON表示的RES ...
- Word2010制作收款单
原文链接:https://www.toutiao.com/i6488255406136099342/ 页面设置 选择"页面布局"选项卡,"页面设置"功能组,&q ...
- 使用.NET 6开发TodoList应用(填坑1)——实现当前登录用户获取
系列导航及源代码 使用.NET 6开发TodoList应用文章索引 需求 在前面的文章里使用.NET 6开发TodoList应用(5)--领域实体创建,我们留了一个坑还没有填上,就是在数据库保存的时候 ...
- 极客大挑战2019 http
极客大挑战 http referer 请求头 xff 1.查看源码,发现secret.php 2.提示要把来源改成Sycsecret.buuoj.cn,抓包,添加Referer Referer:htt ...
- [GKCTF2020]EZ三剑客-EzNode&[GYCTF2020]Ez_Express
写在前面 Nodejs基础一点没有做题还是很难下手的,要学的还很多 [GKCTF2020]EZ三剑客-EzNode 知识点 1.settimeout溢出 2.沙盒逃逸 题解 打开题目,看源代码 app ...
- Javascript——ES6( ECMAScript 6.0)语法
ES6( ECMAScript 6.0)语法 一.let/const与var的区别 var 会进行预解析,let/const不会 var可以声明两个重名的变量,let/const不能 var没有块级作 ...
- Java类与对象的创建
以类的方式组织代码,以对象的方式组织(封装)数据 组织代码(类) public class Demo04 { String name;//默认值null int age;//默认值0 public v ...
- 【程序18】求s=a+aa+aaa+aaaa+aa...a的值
求s=a+aa+aaa+aaaa+aa-a的值,其中a是一个数字.例如2+22+222+2222+22222(此时共有5个数相加),几个数相加由键盘控制. 知识点:在Python 3里,reduce( ...
- 日志模块详细介绍 hashlib模块 动态加盐
目录 一:hashlib模块 二:logging 一:hashlib模块 加密: 将明文数据通过一系列算法变成密文数据(目的就是为了数据的安全) 能够做文件一系列校验 python的hashlib提供 ...
- Arduino+ESP32 之 驱动GC9A01圆形LCD(二),移植LVGL,跑示例程序,显示自制图片
在前文Arduino+ESP32 之 驱动GC9A01圆形LCD(一), 我们已经移植好了arduino GFX库, 该库的示例程序内,还有LVGL的示例程序哦. arduino环境下移植lvgl是很 ...