Solution -「BZOJ #3786」星系探索
Link.
给定一棵含 个点的有根树,点有点权,支持 次操作:
- 询问 到根的点权和;
- 修改 的父亲,保证得到的图仍是树;
- 将 子树内的点权增加一常数。
,。
ETT (Euler Tour Tree),是一种能快速处理子树移动的动态树。本质上,它将树保存作欧拉序,由于子树移动体现在欧拉序上是区间移动,那么就能使用平衡树维护序列。路径查询亦对应了一段区间内仅出现一次的结点贡献和,记每个结点第一次出现为正贡献,后一次出现为负贡献就变成了区间查询,亦能用平衡树维护。本题即 ETT 板题。
复杂度自然是 的。
/*~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;
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· AI与.NET技术实操系列:基于图像分类模型对图像进行分类
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· ollama系列01:轻松3步本地部署deepseek,普通电脑可用
· 25岁的心里话
· 按钮权限的设计及实现
2020-07-08 Solution -「CF 757F」Team Rocket Rises Again
2020-07-08 Solution -「ZJOI2012」「洛谷 P2597」灾难