博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
[科技] 假装是ETT的ETT
阅读量:6680 次
发布时间:2019-06-25

本文共 7136 字,大约阅读时间需要 23 分钟。

[科技] 假装是ETT的ETT

[科技] 假装是ETT的ETT

Codechef 的 April Challenge 2019 Division 1 的 Sonya and Queries 这题的\(45\)分部分分,似乎是一个出栈入栈序\(ETT\),看着似乎还挺好的,就写了写。那么这里就讲一讲这个假装是\(ETT\)\(ETT\)

\(ETT\),全称\(Euler-Tour-Tree\)。实际上真实的\(ETT\)是用树上的欧拉回路来写的,但是也有这种省略欧拉回路为入栈出栈序列的\(ETT\),个人感觉可能后者更容易实现,但是与前者的不同就是似乎后者不能换根,但是可以进行链上操作……如果要学欧拉回路的\(ETT\)的话可以看。

首先出栈入栈序列,或者叫做括号序这个东西,用处还是挺多的,比较常见的就是用在\(RMQ\ LCA\)的问题上,可以做到\(O(nlogn)\)预处理,\(O(1)\)查询\(LCA\)的优秀复杂度。其主要的性质就是对于每一个点的出栈与入栈时间\(in[x]\)\(out[x]\),都有\([in[x], out[x]]\)中所有的节点都是\(x\)的子树中的节点。这个还是比较显然的,这也是\(ETT\)能够在某些地方代替\(LCT\)维护动态树的基础。

于是我们尝试用括号序\(ETT\)实现一些\(LCT\)的操作。

首先是最基础的\(Link / Cut\)

考虑到\(Link / Cut\)就是把\(x\)的子树和\(x\)的父亲断开或者接到另一个节点\(y\)上,而这个操作在括号序上的表现即是序列上一个连续段的平移。这个操作我们再熟悉不过了,利用平衡树就可以优秀地解决了。

然后是一些基础的子树操作,例如子树加,子树赋值,子树求和,子树求\(Max\)等等:

还是考虑括号序的性质,\(x\)的子树都在\([in[x], out[x]]\)这个区间中,在平衡树中将这个区间分裂出来之后找到根节点,就可以得到区间的信息了,或者打一些子树的标记了。不过需要注意的是,在子树求和的时候,由于区间中的数实际上有\(2 * sz\)个,所以答案应该要除以\(2\)

还有一写单点的操作也是和子树操作类似的,就不赘述了。但是有一个细节也是和子树求和差不多的,就是单点操作的时候要操作两个点,这点不要忘记。

然后是一些链上的操作:

(注:这里的链上操作,博主似乎都只会那种直接到根节点的操作,并且这种方法似乎不能求链上\(\min, \max\),如果有什么更为优秀的实现方式,欢迎评论)

考虑括号序的实质是入栈出栈序列,假设我们需要操作是\(x\)到根的一条链,那么我们考虑\([1, out[x]]\)这个区间,如果对于一个节点\(u\)\(in[u]\)\(out[u]\)都在这个区间里的话,那么说明这个节点\(u\)并不在需要操作的链上。
以链求和为例,我们就可以在平衡树插入的节点进行一些修改。对于节点\(x\),我们在\(in[x]\)出插入\(val[x]\),在\(out[x]\)出插入\(-val[x]\)。那么链求和就是\([1, out[x]]\)中的所有节点的和,这个平衡树也是可以优秀的解决的。唯一不怎么优秀的就是需要开两个平衡树……

这些大概就是一些\(ETT\)能够满足的操作了。可能博主学的并不精,勿喷……

给一道例题吧:

代码应该还好写的,就不放出来了……

至于Codechef么……emm……反正比赛也快结束了……\(45\)分的部分分也应该没什么人要的样子啊,我就直接放出来了吧嘿嘿嘿……反正没多少人看……

Code:

用的是\(fhq \ Treap\)来实现的……

大致就是需要支持:\(Link / Cut\)(保证操作的为原树上的边),子树求和,子树赋值,单点加,单点查询,子树加。

还算好写吧……

#include 
using namespace std;const int N = 3E5 + 50;typedef long long ll;typedef pair
P;typedef vector
Vec;#define fi first#define se second#define mk make_pairint n, m, ID, clck;struct edge { int u, v;}E[N << 2];ll a[N];Vec G[N], t;int in[N], out[N];char s[N];namespace Treap { struct node { int ls, rs, sz, zero, fa; ll v, sum, Rnd, tag; }t[N << 1]; int tot, rt; #define ls(o) t[o].ls#define rs(o) t[o].rs#define pa(o) t[o].fa int Newnode(ll V) { t[++tot].ls = t[tot].rs = t[tot].zero = t[tot].fa = t[tot].tag = 0; t[tot].sz = 1; t[tot].Rnd = 1ll * rand() * rand(); t[tot].sum = t[tot].v = V; return tot; } void Update(int o) { t[o].sz = 1; t[o].sum = t[o].v; t[o].sz += ( ls(o) ? t[ls(o)].sz : 0 ) + ( rs(o) ? t[rs(o)].sz : 0 ); t[o].sum += ( ls(o) ? t[ls(o)].sum : 0) + ( rs(o) ? t[rs(o)].sum : 0 ); } void Apply1(int o) { t[o].sum = t[o].v = 0; t[o].zero = 1; t[o].tag = 0; } void Apply2(int o, ll V) { t[o].sum += t[o].sz * V; t[o].v += V; t[o].tag += V; } void Push(int o) { if(t[o].zero) { if(ls(o)) Apply1(ls(o)); if(rs(o)) Apply1(rs(o)); t[o].zero = 0; } if(t[o].tag) { if(ls(o)) Apply2(ls(o), t[o].tag); if(rs(o)) Apply2(rs(o), t[o].tag); t[o].tag = 0; } } int Merge(int x, int y) { if(!x || !y) return x | y; Push(x); Push(y); if(t[x].Rnd < t[y].Rnd) { int k = Merge(t[x].rs, y); t[k].fa = x; t[x].rs = k; Update(x); return x; } else { int k = Merge(x, t[y].ls); t[k].fa = y; t[y].ls = k; Update(y); return y; } } P Split(int o, int k) { if(!o) return mk(0, 0); Push(o); if(t[ls(o)].sz == k) { int Ls = ls(o); ls(o) = 0; t[Ls].fa = 0; Update(o); return mk(Ls, o); } if(t[ls(o)].sz + 1 == k) { int Rs = rs(o); rs(o) = 0; t[Rs].fa = 0; Update(o); return mk(o, Rs); } if(t[ls(o)].sz < k) { P tmp = Split(rs(o), k - t[ls(o)].sz - 1); int Rs = t[o].rs; t[Rs].fa = 0; t[o].rs = tmp.fi; t[tmp.fi].fa = o; Update(o); return mk(o, tmp.se); } P tmp = Split(ls(o), k); int Ls = t[o].ls; t[Ls].fa = 0; t[o].ls = tmp.se; t[tmp.se].fa = o; Update(o); return mk(tmp.fi, o); } P Findroot(int o) { int las = 0, Rnk = ls(o) ? t[ls(o)].sz : 0; while(pa(o)) { if(rs(pa(o)) == o) Rnk += t[ls(pa(o))].sz + 1; o = pa(o); } return mk(o, Rnk); } void Insert(ll V) { int o = Newnode(V); rt = Merge(rt, o); } void Cut(int x, int y) { int L1 = in[x], R1 = out[x], L2 = in[y], R2 = out[y]; if(L1 <= L2 && R2 <= R1) swap(x, y), swap(L1, L2), swap(R1, R2); P L = Findroot(L1), R = Findroot(R1); P tmp1 = Split(L.fi, R.se + 1); P tmp2 = Split(tmp1.fi, L.se); Merge(tmp2.fi, tmp1.se); } void Link(int x, int y) { int L1 = in[x], R1 = out[x], L2 = in[y], R2 = out[y]; if(L1 <= L2 && R2 <= R1) swap(x, y), swap(L1, L2), swap(R1, R2); int o = Findroot(L1).fi; P R = Findroot(R2); P tmp1 = Split(R.fi, R.se + 1); P tmp2 = Split(tmp1.fi, R.se); int p = Merge(tmp2.fi, o); int k = Merge(p, tmp2.se); Merge(k, tmp1.se); }}void Dfs(int o, int fa) { in[o] = ++clck; t.push_back(o); for(int i = 0; i < (int)G[o].size(); i++) { int to = G[o][i]; if(to == fa) continue; Dfs(to, o); } out[o] = ++clck; t.push_back(o);}int main() { Treap::rt = Treap::tot = 0; scanf("%d%d%d", &ID, &n, &m); if(ID == 4 || ID == 5) return puts("nmdwsm"), 0; for(int i = 1, u, v; i < n; i++) { scanf("%d%d", &u, &v); E[i] = (edge) { u, v }; G[u].push_back(v); G[v].push_back(u); } scanf("%s", s + 1); for(int i = 1; i <= n; i++) scanf("%lld", &a[i]); Dfs(1, 1); for(int i = 0; i < (int)t.size(); i++) Treap::Insert(a[t[i]]); for(int i = 1; i < n; i++) { if(s[i] == '0') continue; Treap::Cut(E[i].u, E[i].v); } while(m--) { int tp; scanf("%d", &tp); if(tp == 1) { int x; scanf("%d", &x); if(s[x] == '0') s[x] = '1', Treap::Cut(E[x].u, E[x].v); else s[x] = '0', Treap::Link(E[x].u, E[x].v); } else if(tp == 2) { int x; ll y; scanf("%d%lld", &x, &y); P p = Treap::Findroot(in[x]); Treap::Apply2(p.fi, y); } else if(tp == 3) { int x; scanf("%d", &x); P p = Treap::Findroot(in[x]); ll Sum = Treap::t[p.fi].sum; Sum >>= 1; Treap::Apply1(p.fi); P tmp1 = Treap::Split(p.fi, p.se + 1); P tmp2 = Treap::Split(tmp1.fi, p.se); Treap::t[tmp2.se].sum = Treap::t[tmp2.se].v = Sum; Treap::Merge(Treap::Merge(tmp2.fi, tmp2.se), tmp1.se); p = Treap::Findroot(out[x]); tmp1 = Treap::Split(p.fi, p.se + 1); tmp2 = Treap::Split(tmp1.fi, p.se); Treap::t[tmp2.se].sum = Treap::t[tmp2.se].v = Sum; Treap::Merge(Treap::Merge(tmp2.fi, tmp2.se), tmp1.se); } else if(tp == 4) { int x; scanf("%d", &x); P p = Treap::Findroot(in[x]); P tmp1 = Treap::Split(p.fi, p.se + 1); P tmp2 = Treap::Split(tmp1.fi, p.se); ll ans = Treap::t[tmp2.se].v; printf("%lld\n", ans); Treap::Merge(Treap::Merge(tmp2.fi, tmp2.se), tmp1.se); } else if(tp == 5) { int x; scanf("%d", &x); P p = Treap::Findroot(in[x]); ll ans = Treap::t[p.fi].sum; ans >>= 1; printf("%lld\n", ans); } else if(tp == 6) { int x; scanf("%d", &x); P p = Treap::Findroot(in[x]); Treap::Apply1(p.fi); } } return 0;}

转载于:https://www.cnblogs.com/Apocrypha/p/10701713.html

你可能感兴趣的文章
JS中的闭包
查看>>
DRF(Django Rest Framework)备忘
查看>>
poj3月题解
查看>>
JS省队集训记
查看>>
吴恩达机器学习笔记五_多元分类和神经网络
查看>>
学习 leaflet-2
查看>>
Android ANR 分析解决方法
查看>>
Django框架----权限管理(设计分析以及具体细节)
查看>>
查看tomcat进程,并删除进程
查看>>
Java 测试连接Oracle数据库是否成功,ojdbc7.jar包下载
查看>>
广度优先搜索 BFS
查看>>
boost::program_options
查看>>
javascript中的时间版运动
查看>>
三栏式布局(所谓的圣杯和双飞翼)
查看>>
分享四个经商故事
查看>>
Django基础学习三_路由系统
查看>>
个人使用Viso绘制的简单神经网络实现原理图
查看>>
基于JavaMail的Java邮件发送:简单邮件发送
查看>>
python全栈开发 * 27知识点汇总 * 180710
查看>>
03-02 Java键盘录入
查看>>