\(\mathcal{Description}\)

  Link.

  给定一棵含 \(n\) 个点的有根树,点有点权,支持 \(q\) 次操作:

  1. 询问 \(u\) 到根的点权和;
  2. 修改 \(u\) 的父亲,保证得到的图仍是树;
  3. 将 \(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」星系探索的更多相关文章

  1. Solution -「BZOJ 3812」主旋律

    \(\mathcal{Description}\)   Link.   给定含 \(n\) 个点 \(m\) 条边的简单有向图 \(G=(V,E)\),求 \(H=(V,E'\subseteq E)\ ...

  2. 【bzoj 3786】星系探索

    Description 物理学家小C的研究正遇到某个瓶颈. 他正在研究的是一个星系,这个星系中有n个星球,其中有一个主星球(方便起见我们默认其为1号星球),其余的所有星球均有且仅有一个依赖星球.主星球 ...

  3. Solution -「BZOJ 4316」小C的独立集

    \(\mathcal{Description}\)   Link.   求包含 \(n\) 个结点 \(m\) 条边的仙人掌的最大独立集.   \(n\le5\times10^4\),\(m\le6\ ...

  4. Solution -「BZOJ 3331」压力

    \(\mathcal{Description}\)   Link.   给定一个 \(n\) 个点 \(m\) 条边的连通无向图,并给出 \(q\) 个点对 \((u,v)\),令 \(u\) 到 \ ...

  5. 「BZOJ 4228」Tibbar的后花园

    「BZOJ 4228」Tibbar的后花园 Please contact lydsy2012@163.com! 警告 解题思路 可以证明最终的图中所有点的度数都 \(< 3\) ,且不存在环长是 ...

  6. 「BZOJ 3645」小朋友与二叉树

    「BZOJ 3645」小朋友与二叉树 解题思路 令 \(G(x)\) 为关于可选大小集合的生成函数,即 \[ G(x)=\sum[i\in c ] x^i \] 令 \(F(x)\) 第 \(n\) ...

  7. 「BZOJ 4502」串

    「BZOJ 4502」串 题目描述 兔子们在玩字符串的游戏.首先,它们拿出了一个字符串集合 \(S\),然后它们定义一个字符串为"好"的,当且仅当它可以被分成非空的两段,其中每一段 ...

  8. 「BZOJ 4289」 PA2012 Tax

    「BZOJ 4289」 PA2012 Tax 题目描述 给出一个 \(N\) 个点 \(M\) 条边的无向图,经过一个点的代价是进入和离开这个点的两条边的边权的较大值,求从起点 \(1\) 到点 \( ...

  9. 「BZOJ 2534」 L - gap字符串

    「BZOJ 2534」 L - gap字符串 题目描述 有一种形如 \(uv u\) 形式的字符串,其中 \(u\) 是非空字符串,且 \(v\) 的长度正好为 \(L\), 那么称这个字符串为 \( ...

随机推荐

  1. [ vue ] xxxProject项目杂记

    2020.4.9 加入eCharts 2020.4.8 完成article的显示,其间碰到全局路由守卫写的有错误,导致跳转报错.已修复. 加入keep-alive功能,缓存视图数据 疑问:如果在全局组 ...

  2. [ bootstrap ] 实现卡片里面包含图片、内容、操作按钮,形成左中右的布局

    描述: 如图 实现: <div class="card border-0 mb-3 mt-3"> <!-- 列表头部 --> <div class=& ...

  3. Python爬取中国知网文献、参考文献、引证文献

    前两天老师派了个活,让下载知网上根据高级搜索得到的来源文献的参考文献及引证文献数据,网上找了一些相关博客,感觉都不太合适,因此特此记录,希望对需要的人有帮助. 切入正题,先说这次需求,高级搜索,根据中 ...

  4. 兼容h5在ios上登录窗口input失焦后页面不会滚

    $("#logincode").blur(function(){ //滚动到顶部 window.scrollTo(0, 0); })一般多用于密码框失焦后,页面不回滚

  5. 【Java】反射

    文章目录 反射 概述 动态语言与非动态语言 动态语言 非动态语言 Java反射机制提供的功能 反射相关的主要API 关于java.lang.Class类的理解 类的加载过程 获取Class的实例的方式 ...

  6. 《剑指offer》面试题26. 树的子结构

    问题描述 输入两棵二叉树A和B,判断B是不是A的子结构.(约定空树不是任意一个树的子结构) B是A的子结构, 即 A中有出现和B相同的结构和节点值. 例如: 给定的树 A:      3     / ...

  7. vue组件实现图片的拖拽和缩放

    vue实现一个组件其实很简单但是要写出一个好的可复用的组件那就需要多学习和钻研一下,一个好的组件必须有其必不可少的有优点:一是能提高应用开发效率.测试性.复用性等:二是组件应该是高内聚.低耦合的:三是 ...

  8. Android学习笔记3

    (5)练习做一个实现两个数相乘的APP ①.java代码: //MainActivity.java package com.example.hello; import android.content. ...

  9. 【机器学习】HMM

    机器学习算法-HMM 目录 机器学习算法-HMM 1. 模型定义 2. 序列生成 3. 概率计算 3.1 前向计算 3.2 后向计算 4. 学习 4.1 求解 4.2 求解 4.3 求解 5. 预测 ...

  10. python -m详解

    温馨提示: 本篇演示环境是Python 3.8 先python --help看下python -m参数的解释: -m mod : run library module as a script (ter ...