传送门

数据,标程

题意:

一颗有根树,支持询问点到根路径权值和,子树加,换父亲


欧拉序列怎么求路径权值和?

一个点的权值只会给自己的子树中的点贡献入栈权值正出栈权值负,求前缀和就行了!

和上题一样,伪ETT大法好

注意本题的子树需要根,所以需要找到子树区间左右的前驱和后继节点把他们splay出来才能得到子树区间,不能直接$l-1, r+1$,一开始写错了

然后注意下放标记,splay需要记录splay子树里有几个入栈几个出栈

加强版:询问任意一条路径$(u,v)$

当然需要减去lca了,于是用一个LCT维护树的形态

#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
using namespace std;
#define lc t[x].ch[0]
#define rc t[x].ch[1]
#define pa t[x].fa
#define pii pair<int, int>
#define MP make_pair
#define fir first
#define sec second
typedef long long ll;
const int N=2e5+;
inline int read(){
char c=getchar();int x=,f=;
while(c<''||c>''){if(c=='-')f=-; c=getchar();}
while(c>=''&&c<=''){x=x*+c-''; c=getchar();}
return x*f;
} int n, a[N], x, y, Q, fa[N]; char s[];
struct edge{int v, ne;} e[N];
int cnt, h[N];
inline void ins(int u, int v) {
e[++cnt]=(edge){v, h[u]}; h[u]=cnt;
}
pii dfn[N]; int dfc, eul[N];
void dfs(int u) {
dfn[u].fir = ++dfc; eul[dfc] = u;
for(int i=h[u]; i; i=e[i].ne) dfs(e[i].v);
dfn[u].sec = ++dfc; eul[dfc] = -u;
} struct meow{int ch[], fa, v, sl, sr, type; ll sum, tag;} t[N];
int root, sz;
inline int wh(int x) {return t[pa].ch[] == x;}
inline void update(int x) {
t[x].sum = t[lc].sum + t[rc].sum + t[x].v;
t[x].sl = t[lc].sl + t[rc].sl + (t[x].type==);
t[x].sr = t[lc].sr + t[rc].sr + (t[x].type==-);
} inline void paint(int x, int d) {
t[x].sum += (ll)d*(t[x].sl - t[x].sr);
t[x].v += d*t[x].type;
t[x].tag += d;
}
inline void pushDown(int x) { //printf("pushDown %d\n",x);
if(t[x].tag) { //puts("yes");
if(lc) paint(lc, t[x].tag);
if(rc) paint(rc, t[x].tag);
t[x].tag = ;
}
}
void pd(int x) { if(pa) pd(pa); pushDown(x); } inline void rotate(int x) {
int f=t[x].fa, g=t[f].fa, c=wh(x);
if(g) t[g].ch[wh(f)] = x; t[x].fa=g;
t[f].ch[c] = t[x].ch[c^]; t[t[f].ch[c]].fa=f;
t[x].ch[c^] = f; t[f].fa=x;
update(f); update(x);
}
inline void splay(int x, int tar) {
pd(x);
for(; pa!=tar; rotate(x))
if(t[pa].fa != tar) rotate(wh(x)==wh(pa) ? pa : x);
if(tar==) root=x;
} void build(int &x, int l, int r, int f) {
int mid = (l+r)>>; x=mid;
t[x].fa=f;
if(eul[x]>) t[x].type = , t[x].v = a[eul[x]];
else t[x].type = -, t[x].v = -a[-eul[x]];
if(l<mid) build(lc, l, mid-, x);
if(mid<r) build(rc, mid+, r, x);
update(x);
} inline int pre(int x) {
x = lc; while(rc) x = rc; return x;
}
inline int nex(int x) {
x = rc; while(lc) x = lc; return x;
}
inline void Split(int &p, int &x) {
splay(p, ); p = pre(p);
splay(x, ); x = nex(x);
splay(p, ); splay(x, p);
} ll Que(int u) {
int p = dfn[].fir, x = dfn[u].fir;
Split(p, x);
return t[lc].sum;
} void Cha(int u, int far) {
int p = dfn[u].fir, x = dfn[u].sec;
Split(p, x);
int q = lc;
lc = t[q].fa = ;
update(x); update(p); p = dfn[far].fir; splay(p, );
x = nex(p); splay(x, p);
lc = q; t[q].fa = x;
update(x); update(p);
} void AddVal(int u, int d) {
int p = dfn[u].fir, x = dfn[u].sec;
Split(p, x);
paint(lc, d);
} int main() {
//freopen("in","r",stdin);
freopen("galaxy.in", "r", stdin);
freopen("galaxy.out", "w", stdout);
n=read();
for(int i=; i<=n; i++) ins(read(), i);
for(int i=; i<=n; i++) a[i]=read();
dfc=; dfs(); dfc++;
build(root, , dfc, );
Q=read();
for(int i=; i<=Q; i++) {
scanf("%s", s); x=read();
if(s[]=='Q') printf("%lld\n", Que(x));
else if(s[]=='C') Cha(x, read());
else AddVal(x, read());
}
}

BZOJ 3786: 星系探索 [伪ETT]的更多相关文章

  1. [BZOJ3786]星系探索(伪ETT)

    3786: 星系探索 Time Limit: 40 Sec  Memory Limit: 256 MBSubmit: 1638  Solved: 506[Submit][Status][Discuss ...

  2. BZOJ 3786: 星系探索 解题报告

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

  3. BZOJ 3786: 星系探索 ETT

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

  4. bzoj 3786 星系探索 dfs+splay

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

  5. BZOJ 3786 星系探索

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

  6. BZOJ 3786 星系探索 ——Splay

    子树可以移动,唔. 还是用Splay维护DFS序即可. 子树的话直接截取出来就好了. 然后求前驱后继可能麻烦一些. 添加两个虚拟节点会比较好写. #include <map> #inclu ...

  7. BZOJ 3786 星系探索 (splay+dfs序)

    题目大意:给你一棵树,支持一下三种操作 1.获取某节点到根节点的路径上所有节点的权值和 2.更换某棵子树的父亲 3.某子树内所有节点的权值都增加一个值w 当时想到了splay维护dfs序,查完题解发现 ...

  8. BZOJ 3786: 星系探索 欧拉游览树

    一个叫 Euler-Tour-Tree 的数据结构,说白了就是用 Splay_Tree 维护欧拉序 #include <cstring> #include <algorithm> ...

  9. 【BZOJ】3786: 星系探索

    [题意]给定一棵带点权树,三种操作: 1.询问点x到根的路径和 2.子树x内的点权加定值y 3.将点x的父亲更换为y,保证仍是树. [算法]平衡树(fhq-treap) [题解] 将树的dfs序作为序 ...

随机推荐

  1. 最小生成树&最短路基础算法总结

    [最短路问题] 解决最短路问题有以下算法:Dijkstra算法,Bellman-Ford算法,Floyd算法,和SPFA算法和启发式搜索算法A*; 每个算法都有它的特点可以解决某些特定的问题,例如:F ...

  2. Monthly update for Dynamics 365 for Operation

    日期 标题, 类别 版本 描述 2017/8/22 Dyn 365 Fin and Ops, Ent ed July 2017 Plat Update 10 Category: Download   ...

  3. 高可用高性能分布式文件系统FastDFS实践Java程序

    在前篇 高可用高性能分布式文件系统FastDFS进阶keepalived+nginx对多tracker进行高可用热备 中已介绍搭建高可用的分布式文件系统架构. 那怎么在程序中调用,其实网上有很多栗子, ...

  4. SSH中后台传到前台一个信息集合,tr td中怎么进行排列,类似在一个div里排列书籍

    总觉得描述问题不对,这里详细说一下,就是把下面图片变成排列整齐,一行四个,多出来的两个排到下一行. 我问过群里的,给的答案都有些简介:1:后台排好了,前台循环出来: 2:前台直接循环,多出来的加< ...

  5. JS URI Encode

    javascript中存在几种对URL字符串进行编码的方法:escape/encodeURI/encodeURIComponent.这几种编码所起的作用各不相同. escape 采用ISO Latin ...

  6. ThinkPHP3.2 实现Mysql数据库备份

    <?php header("Content-type:text/html;charset=utf-8"); //配置信息 $cfg_dbhost = 'localhost'; ...

  7. Thinkphp+Nginx(PHPstudy)下报的404错误,403错误解决

    最近一个TP5的项目说放到Nginx下测试看看,下载个 PHPstudy,放到WWW下,配置好域名,直接给个报个404: 解决方法: 1.先在phpstudy下配置好域名目录指向项目下的public下 ...

  8. CentOS系统下Redis安装和自启动配置的步骤

    相信大家都知道Redis是一个C实现的基于内存.可持久化的键值对数据库,在分布式服务中常作为缓存服务.所以这篇文章将详细介绍在CentOS系统下如何从零开始安装到配置启动服务.有需要的可以参考借鉴. ...

  9. arclistsg文档独立模型标签

    [标签名称] arclistsg [标签简介] 单表独立模型的文档列表调用标记 [功能说明] 用于调用单表模型的内容,在V5.3系统以上版本中加入了单表模型的概念,脱离了以前的主从表的数据表关联结构, ...

  10. 数据库DBUtils基本使用

    Apache-DBUtils简介 commons-dbutils 是 Apache 组织提供的一个开源 JDBC工具类库,它是对JDBC的简单封装,使用dbutils能极大简化jdbc编码的工作量,同 ...