【学习笔记】浅谈短小可爱的左偏树(可并堆)

文章目录

  • 左偏树
  • 左偏树的合并(merge)操作
  • 例题
    • 罗马游戏
    • [Apio2012]dispatching
    • [JLOI2015]城池攻占
    • [Baltic2004]sequence

左偏树

左偏树是一个堆,而且是一个可并堆,所以一定有权值的限制

以小根堆为例,那么必须满足节点权值小于左右儿子权值,即val[i]<val[lson[i]] val[i]<val[rson[i]]

为了维护左偏树的时间复杂度,就要设计一些左偏树独有的工具,定义

  • 外节点:满足左右儿子至少有一个没有的点
  • 距离dis[i]:点iii到离它最近的一个外节点的距离;特别的,空节点的距离为−1-11,外节点本身距离为000

左偏树,树如其名,向左偏的树

也就是说左边要“重一点”,但这个重不是常见的子树大小或者高度,而是我们定义的dis

  • 对于左偏树的每个点iii,满足dis[lson[i]]>=dis[rson[i]]

完全有可能,左儿子只有一个点,右儿子是一条链:不管是左儿子的一个点还是右儿子一条链上的点都符合外节点定义,所以dis都是000

考虑由下向上更新距离数组,即dis[i]=min(dis[lson[i]],dis[rson[i]])+1

由于左偏树的性质,右儿子距离一定是小的,所以左偏树的更新直接就是从右儿子处更新

  • dis[i]=dis[rson[i]]+1

左偏树特殊距离的定义,就保证了左偏树的时间复杂度

  • 节点数为nnn的树,树根的dis≤log⁡(n+1)−1dis\le \log(n+1)-1dislog(n+1)1

一个点的距离大于xxx,意味着x−1x-1x1层内都是满二叉树

左偏树的合并(merge)操作

左偏树最重要的,或者说特色,就是它的合并操作

有了以上的理论基础,结合fhq-treap的返回根合并操作,左偏树的合并还是很好写的

struct node { int l, r, val, dis; } t[maxn];
int merge( int x, int y ) {if( ! x or ! y ) return x + y;if( t[x].val > t[y].val ) swap( x, y );t[x].r = merge( t[x].r, y );if( t[t[x].l].dis < t[t[x].r].dis ) swap( t[x].l, t[x].r );t[x].dis = t[t[x].r].dis + 1;return x;
}
int lson[maxn], rson[maxn], val[maxn], dis[maxn];
int merge( int x, int y ) {if( ! x or ! y ) return x + y;if( val[x] < val[y] ) swap( x, y );rson[x] = merge( rson[x], y );if( dis[lson[x]] < dis[rson[x]] ) swap( lson[x], rson[x] );dis[x] = dis[rson[x]] + 1;return x;
}

例题

罗马游戏

luogu 2713

这就是比较板的题了,初始时,把每个士兵当成一个左偏树,合并就是两个士兵的左偏树合并

杀士兵,就是去掉左偏树的根节点,直接合并左右儿子即可

但是怎么找士兵所在的兵团呢?——并查集!

#include <cstdio>
#include <iostream>
using namespace std;
#define maxn 1000005
struct node { int l, r, val, dis; } t[maxn];
bool vis[maxn];
int f[maxn];
int n, m;int find( int x ) { return x == f[x] ? x : f[x] = find( f[x] ); }int merge( int x, int y ) {if( ! x or ! y ) return x + y;if( t[x].val > t[y].val ) swap( x, y );t[x].r = merge( t[x].r, y );if( t[t[x].l].dis < t[t[x].r].dis ) swap( t[x].l, t[x].r );t[x].dis = t[t[x].r].dis + 1;return x;
}int main() {scanf( "%d", &n );t[0].dis = -1;for( int i = 1;i <= n;i ++ ) {scanf( "%d", &t[i].val );f[i] = i;}scanf( "%d", &m );char opt[5]; int i, j;while( m -- ) {scanf( "%s", opt );if( opt[0] == 'M' ) {scanf( "%d %d", &i, &j );if( vis[i] or vis[j] ) continue;i = find( i ), j = find( j );if( i ^ j ) f[j] = f[i] = merge( i, j );}else {scanf( "%d", &i );if( vis[i] ) printf( "0\n" );else {i = find( i );vis[i] = 1;printf( "%d\n", t[i].val );f[i] = f[t[i].l] = f[t[i].r] = merge( t[i].l, t[i].r );/*就算杀了根节点代表的这个士兵也要更新士兵的并查集集合因为可能有些士兵跨过了根节点的左右儿子直接并查集是接在根节点并查集的下面的find的时候是会跳到根节点的并查集点的*/}}}return 0;
}

[Apio2012]dispatching

BZOJ 2809

dfs树从下往上合并

每次丢掉最大的CiC_iCi即可

同样别忘了并查集

#include <cstdio>
#include <vector>
#include <iostream>
using namespace std;
#define maxn 100005
#define int long long
vector < int > G[maxn];
int lson[maxn], rson[maxn], dis[maxn], f[maxn], L[maxn], C[maxn];
int n, m, root;
long long ans;int find( int x ) { return f[x] == x ? x : f[x] = find( f[x] ); }int merge( int x, int y ) {if( ! x or ! y ) return x + y;if( C[x] < C[y] ) swap( x, y );rson[x] = merge( rson[x], y );if( dis[lson[x]] < dis[rson[x]] ) swap( lson[x], rson[x] );dis[x] = dis[rson[x]] + 1;return x;
}pair < int, int > dfs( int u ) {int cnt = 1, sum = C[u];for( auto v : G[u] ) {pair < int, int > son = dfs( v );cnt += son.first, sum += son.second;int x = find( u ), y = find( v );root = f[u] = f[v] = merge( x, y );while( sum > m ) {sum -= C[root], cnt --;f[root] = f[lson[root]] = f[rson[root]] = merge( lson[root], rson[root] );root = f[root];}ans = max( ans, L[u] * cnt );}ans = max( ans, L[u] * cnt );return make_pair( cnt, sum );
}signed main() {scanf( "%lld %lld", &n, &m );for( int i = 1, B;i <= n;i ++ ) {scanf( "%lld %lld %lld", &B, &C[i], &L[i] );G[B].push_back( i );f[i] = i;}dis[0] = -1;dfs( 1 );printf( "%lld\n", ans );return 0;
}

[JLOI2015]城池攻占

luogu 3261

初始时,将每一座城市当成一个左偏树,并把在该城市的士兵在左偏树上完成合并

与上一题一样的,从下往上合并左偏树,把所有儿子的骑士聚集到当前根节点城市

把战斗力不足的扔掉,所以需要小根堆

利用初末城市的深度差,巧妙计算出每个骑士能攻占的城市数量

至于城市对骑士的战斗力影响,一样的整体懒标记,碰到了再下传

#include <cstdio>
#include <vector>
#include <iostream>
using namespace std;
#define int long long
#define maxn 300005
vector < int > G[maxn];
struct node { int lson, rson, dis, val, add, mul; }t[maxn];
int root[maxn], h[maxn], a[maxn], v[maxn], c[maxn];
int ans1[maxn], ans2[maxn], dep[maxn];
int n, m;void modify( int x, int mul, int add ) {if( ! x ) return;t[x].val *= mul;t[x].val += add;t[x].mul *= mul;t[x].add *= mul;t[x].add += add;
}void pushdown( int x ) { if( ! x ) return;modify( t[x].lson, t[x].mul, t[x].add );modify( t[x].rson, t[x].mul, t[x].add );t[x].mul = 1;t[x].add = 0;
}int merge( int x, int y ) {if( ! x or ! y ) return x + y;pushdown( x );pushdown( y );if( t[x].val > t[y].val ) swap( x, y );t[x].rson = merge( t[x].rson, y );if( t[t[x].lson].dis < t[t[x].rson].dis ) swap( t[x].lson, t[x].rson );t[x].dis = t[t[x].rson].dis + 1;return x;
}void dfs( int u, int fa ) {dep[u] = dep[fa] + 1;for( auto v : G[u] ) {dfs( v, u );root[u] = merge( root[u], root[v] );//将所有从子树的管辖城市中存活到u城市的骑士进行合并 }while( root[u] and t[root[u]].val < h[u] ) {//从最小杀伤力骑士开始 将所有在u城市死亡的骑士剔除 并记录最后答案 pushdown( root[u] );ans1[u] ++;ans2[root[u]] = dep[c[root[u]]] - dep[u];//骑士能攻占的数量巧妙利用初末深度差 最开始的城市深度-u的深度 root[u] = merge( t[root[u]].lson, t[root[u]].rson ); }if( a[u] ) modify( root[u], v[u], 0 );else modify( root[u], 1, v[u] );
}signed main() {scanf( "%lld %lld", &n, &m );for( int i = 1;i <= n;i ++ ) scanf( "%lld", &h[i] );for( int i = 2, f;i <= n;i ++ ) {scanf( "%lld %lld %lld", &f, &a[i], &v[i] );G[f].push_back( i );}t[0].dis = -1;//在每一座城市都建立一个左偏树 for( int i = 1;i <= m;i ++ ) {scanf( "%lld %lld", &t[i].val, &c[i] );root[c[i]] = merge( root[c[i]], i );//将在同一座城市的骑士在左偏树上体现合并 //root[i]:在城市i的左偏树的最小杀伤力骑士的编号 }dfs( 1, 0 );//搜城 从下往上爬 while( root[1] ) {/*最后在总城市根1的骑士答案还未计算 暴力弹出所有存活的骑士 攻占的数量就是初始城市到1号超级城市一路经过的城市也就是深度*/ ans2[root[1]] = dep[c[root[1]]];root[1] = merge( t[root[1]].lson, t[root[1]].rson );}for( int i = 1;i <= n;i ++ ) printf( "%lld\n", ans1[i] );for( int i = 1;i <= m;i ++ ) printf( "%lld\n", ans2[i] );return 0;
}

[Baltic2004]sequence

BZOJ 1367

给定nnn,以及长度为nnn的整数序列a1,...,ana_1,...,a_na1,...,an,你需要构造一个严格上升的长度为nnn整数序列t1,...,tnt_1,...,t_nt1,...,tn,定义R=∣t1−a1∣+...+∣ti−ai∣+...+∣tn−an∣R=|t_1-a_1|+...+|t_i-a_i|+...+|t_n-a_n|R=t1a1+...+tiai+...+tnan,求最小的RRR

n≤106n\le 10^6n106


假设是构造不下降序列

考虑两种特殊的序列

  • a1≤a2≤...≤ana_1\le a_2\le ...\le a_na1a2...an,直接让ti=ait_i=a_iti=ai,最后答案就是000
  • a1≥a2≥...≥ana_1\ge a_2\ge ...\ge a_na1a2...an,直接让t1=amidt_1=a_{mid}t1=amid(序列的中位数)

现在考虑一般形式的aaa

aaa序列划分成这两种特殊序列的若干段

考虑1...i1...i1...i的答案为w1,...wiw_1,...w_iw1,...wi,现在考虑i+1i+1i+1,先让wi+1=ai+1w_{i+1}=a_{i+1}wi+1=ai+1

然后就要根据第二种特殊序列,进行合并,如果wi≥wi+1w_i\ge w_{i+1}wiwi+1,就合并后wi=wi+1=w_i=w_{i+1}=wi=wi+1=中位数,直到wj<wi+1w_j<w_{i+1}wj<wi+1

这个中位数是参与合并的www中的中位数,不是整体的中位数

用左偏树维护一个sizsizsiz大小即可

#include <cstdio>
#include <iostream>
using namespace std;
#define maxn 1000005
struct node {int lson, rson, val, siz, dis;
}t[maxn];
int n, cnt;
int root[maxn], L[maxn], R[maxn], a[maxn];int merge( int x, int y ) {if( ! x or ! y ) return x + y;if( t[x].val < t[y].val ) swap( x, y );t[x].rson = merge( t[x].rson, y );t[x].siz = t[t[x].lson].siz + t[t[x].rson].siz + 1;if( t[t[x].lson].dis < t[t[x].rson].dis )swap( t[x].lson, t[x].rson );t[x].dis = t[t[x].rson].dis + 1;return x;
}long long Fabs( long long x ) {return x < 0 ? -x : x;
}int NewNode( int x ) {t[++ cnt].val = x;t[cnt].siz = 1;t[cnt].lson = t[cnt].rson = t[cnt].dis = 0;return cnt;
}int main() {scanf( "%d", &n );for( int i = 1;i <= n;i ++ )scanf( "%d", &a[i] ), a[i] -= i;t[0].dis = -1;int ip = 0;for( int i = 1;i <= n;i ++ ) {++ ip;L[ip] = R[ip] = i;root[ip] = NewNode( a[i] );while( ip > 1 and t[root[ip - 1]].val > t[root[ip]].val ) {root[ip - 1] = merge( root[ip - 1], root[ip] );R[ip - 1] = R[ip];ip --;while( t[root[ip]].siz * 2 > R[ip] - L[ip] + 2 )root[ip] = merge( t[root[ip]].lson, t[root[ip]].rson );}}long long ans = 0;while( ip ) {int val = t[root[ip]].val;for( int i = L[ip];i <= R[ip];i ++ )ans += Fabs( 1ll * val - a[i] );ip --;}printf( "%lld\n", ans );return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/317001.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

2.12 模拟

文章目录前言题目解析染色计划&#xff08;color&#xff09;奇度边集&#xff08;edges&#xff09;猜拳游戏&#xff08;guess&#xff09;代码T1T2LCT整体二分总结前言 120pts 期望&#xff1a;4010020160 实际&#xff1a;406020120 rnk 9 我yue了。 怎么又是不可抗力性挂…

Acwing1069. 凸多边形的划分

Acwing1069. 凸多边形的划分 题意&#xff1a; 一个N个顶点的凸多边形&#xff0c;划分成N-2个互不相交的三角形&#xff0c;对于每个三角形&#xff0c;其三个顶点的权值相乘都可得到一个权值乘积&#xff0c;试求所有三角形的顶点权值乘积之和至少为多少。 题解&#xff1…

徐磊(Devops):一名写了十几年代码但还没写够的程序员

徐磊&#xff08;Devops 社区领袖&#xff09;【个人介绍】徐磊&#xff0c;微软MVP&#xff08;微软最有价值专家&#xff0c;大中华区域社区技术总监&#xff0c;Devops 社区领袖&#xff09;&#xff0c;从事过网管、技术支持、网络、软件开发等工作&#xff0c;一名写了十几…

加强版[BZOJ#3483] SGU505 Prefixes and suffixes(询问在线版)

文章目录descriptionsolutioncode#3483. SGU505 Prefixes and suffixes&#xff08;询问在线版&#xff09;description 给定&#x1d45b;个字符串&#xff0c;有&#x1d45a;个询问。 每个询问给出两个字符串&#x1d460;1, &#x1d460;2&#xff0c;问&#x1d45b;个字…

AcWing 320. 能量项链

AcWing 320. 能量项链 题意&#xff1a; 题解&#xff1a; 和环形石头合并基本一样 代码&#xff1a; #include<bits/stdc.h> #define debug(a,b) printf("%s %d\n",a,b); typedef long long ll; using namespace std;inline int read(){int s0,w1;char c…

ERP不规范,同事两行泪

最近的很多次对外交流&#xff0c;都聊到了ERP建设的话题&#xff0c;并且无一例外的不那么让人省心&#xff0c;回想我这么多年走过的ERP坑坑路&#xff0c;在这里也写下经验和总结&#xff0c;希望能给正在或者即将走上ERP建设路的企业一些思考和帮助。导读1、几个瞎眼而普遍…

2.13模拟总结

文章目录前言题目解析最小划分&#xff08;divide&#xff09;进制路径&#xff08;base&#xff09;欧拉欧拉&#xff08;eular&#xff09;代码T1T2T3总结前言 day9 170pts 期望&#xff1a;10010020220 实际&#xff1a;701000170 rnk7 挂的分有点多qwq 分数要是得满就能拿…

Panasonic Programming Contest (AtCoder Beginner Contest 195) 题解

文章目录A - Health M DeathB - Many OrangesC - CommaD - Shipping CenterE - Lucky 7 BattleF - Coprime PresentPanasonic Programming Contest (AtCoder Beginner Contest 195)A - Health M Death 判断倍数。 #include <cstdio> int main() {int M, H;scanf( "…

长沙.NET社区之光

奈何万事开头难迎着改革开放四十年带来的春风&#xff0c;长沙的互联网生态环境以唐胡子俱乐部为首的一众互联网社群将长沙互联网的环境推上了一个新的台阶。年底&#xff0c;我与有幸一起共事的溪源兄&#xff0c;下班后一起闲聊&#xff0c;觉着长沙的.NET的生态环境亟待改善…

P3205 [HNOI2010]合唱队

P3205 [HNOI2010]合唱队 题意&#xff1a; 有n个数&#xff0c;然后插入队伍中&#xff0c;如果队列当前为空&#xff0c;则直接插入&#xff0c;然后每次插入和上一次插入的比较&#xff0c;如果大于&#xff0c;插入当前队列的最右侧&#xff0c;如果小于&#xff0c;插入当…

2.14模拟总结

前言 节日快乐&#xff01; (逃) day10 50pts 期望&#xff1a;10302060 实际&#xff1a;0302050 rnk11 彻彻底底的摆烂局了。 但是rnk竟然没有太掉&#xff0c;所以我应该并不孤独… 和KH并排坐在机房里&#xff0c;各自看着电脑&#xff0c;痴痴想着各自的心事&#xff0c;…

KYOCERA Programming Contest 2021(AtCoder Beginner Contest 200)题解

文章目录A - CenturyB - 200th ABC-200C - Ringos Favorite Numbers 2D - Happy Birthday! 2E - Patisserie ABC 2F - Minflip SummationKYOCERA Programming Contest 2021&#xff08;AtCoder Beginner Contest 200&#xff09; A - Century 简单的除以200200200向上取整 B…

高级进阶:Azure DevOps搞定.NET Core编译版本号自增

点击上方蓝字关注“汪宇杰博客”熟悉.NET Framework的人知道&#xff0c;我们可以通过指定AssemblyVersion为10.0.*来让编译器自增版本号。但是.NET Core和.NET Standard不行。即使有MSBump这样的开源项目&#xff0c;也有一定的缺陷。一般这样的需求会出现在CI/CD服务器上。我…

Little Boxes UVALive - 8209

Little Boxes UVALive - 8209 题意&#xff1a; 给你四个数&#xff0c;输出四个数之和&#xff0c;四个数小于等于262之内 题解&#xff1a; 这。。。这。。水题 unsigned int 0&#xff5e;4294967295 (10位数&#xff0c;4e9) int -2147483648&#xff5e;2147483647 (…

2.15模拟总结

前言 day11 期望&#xff1a;406030130 实际&#xff1a;4003070 rnk16 挂大分了。。 T2树边不加双向&#xff1a;60->0。 这什么伞兵bug啊&#xff01; 整体状态也不太好&#xff0c;T2死磕无果。 T1看出正解结果写不出来拉插&#xff0c;乐。 题目解析 T1 网格序列&am…

Mynavi Programming Contest 2021(AtCoder Beginner Contest 201)题解

文章目录A - Tiny Arithmetic SequenceB - Do you know the second highest mountain?C - Secret NumberD - Game in Momotetsu WorldE - Xor DistancesF - Insertion SortMynavi Programming Contest 2021&#xff08;AtCoder Beginner Contest 201&#xff09;A - Tiny Arit…

Newbe.Claptrap - 一套以 “事件溯源” 和“Actor 模式”作为基本理论的服务端开发框架...

本文是关于 Newbe.Claptrap 项目主体内容的介绍&#xff0c;读者可以通过这篇文章&#xff0c;大体了解项目内容。轮子源于需求随着互联网应用的蓬勃发展&#xff0c;相关的技术理论和实现手段也在被不断创造出来。诸如 “云原生架构”、“微服务架构”、“DevOps” 等一系列关…

Rabbits UVALive - 8211

Rabbits UVALive - 8211 题意&#xff1a; n个兔子的位置&#xff0c;兔子每次可以跳到两个兔子之间&#xff0c;问最多可以跳多少下&#xff1f; 题解&#xff1a; 求出所有相邻两数的间隔&#xff0c;然后减去最小间隔就是答案 代码&#xff1a; #include <bits/std…

2.16模拟总结

前言 期望&#xff1a;100700170 实际&#xff1a;400040 rnk14 分全部挂没了&#xff0c;太行了。 T1不开longlong见祖宗&#xff0c;而且KH说的那个也有道理&#xff0c;带权之后树的重心可以不只有两个&#xff0c;所以最后还应该倍增的跳。&#xff08;然而这个地方题解似…

Caddi Programming Contest 2021(AtCoder Beginner Contest 193) 题解

Caddi Programming Contest 2021(AtCoder Beginner Contest 193) A - Discount 打折浮点数除即可 B - Play Snuke 枚举判断符合要求的求最小值即可 C - Unexpressed O(n)O(\sqrt{n})O(n​)枚举aaa&#xff0c;暴力翻倍&#xff08;最小的222最多乘323232次就会超过nnn的上…