昨天看了可并堆是什么,写的是左偏树

大概就是一棵树

1、有左偏性质,即当前根到左叶子节点距离比到右叶子节点距离大

2、有堆性质,堆顶关键字比子树关键字小

合并两个堆的时候,关键字大的插入到关键字小的那堆的右子树中,右子树的深度大于左子树时交换两者以维持左偏性质。

堆中个数太多的时候,pop堆顶的元素x

具体是合并左右两个子树后返回新的堆顶元素y,清除完x的信息后root[x]指向y

bzoj1455:用并查集合并团,用可并堆维护团中最小的人。

 #include<stdio.h>
 #include<string.h>
 #include<algorithm>
 using namespace std;
 ;
 int n,m,v[maxn],vis[maxn],fa[maxn],l[maxn],r[maxn],a,b,d[maxn];
 ];

 int find(int x){
     return (fa[x]==x)?x:fa[x]=find(fa[x]);
 }

 int merge(int x, int y){
     if (!x || !y) return x+y;
     if (v[x]>v[y]) swap(x,y);  //保证堆顶最小
     r[x]=merge(r[x],y);
     if (d[r[x]]>d[l[x]]) swap(l[x],r[x]);
     d[x]=d[r[x]]+;
     return x;
 }

 int main(){
     scanf("%d", &n);
     ; i<=n; i++) scanf("%d", &v[i]);
     scanf(]=-;
     ; i<=n; i++) fa[i]=i;
     ; i<=m; i++){
         scanf("%s", s);
         ]=='M'){
             scanf("%d%d", &a, &b);
             if (vis[a] || vis[b]) continue;
             int fx=find(a), fy=find(b);
             if (fx!=fy) fa[fx]=fa[fy]=merge(fx,fy); // 合并从堆顶开始合并
         }else{
             scanf("%d", &a);
             ");
             else{
                 ;// printf("  %d\n", fx);
                 printf("%d\n", v[fx]);
                 fa[fx]=merge(l[fx],r[fx]);
                 fa[fa[fx]]=fa[fx];
             }
         }
     }
     ;
 }

bzoj2809:DFS一遍,从叶子节点开始向上递归,到一个节点,可并堆维护当前子树中满足预算的最多人数,然后用当前子树更新答案。具体是每个节点都插入,超出预算就pop最大花费的人直到满足预算

 #include<stdio.h>
 #include<string.h>
 #include<algorithm>
 #define LL long long
 using namespace std;
 ;
 struct node{
     int to,next;
 }e[maxn];
 struct heap{
     int l,r;
     LL val;
 }t[maxn];
 ,head[maxn];
 LL v[maxn],w[maxn],ans,sum[maxn],add[maxn];

 void insert(int u, int v){
     e[++tot].to=v; e[tot].next=head[u]; head[u]=tot;
 }

 int merge(int x, int y){
     if (!x || !y) return x+y;
     if (t[x].val<t[y].val) swap(x,y);
     t[x].l=merge(t[x].l,y);
     swap(t[x].l,t[x].r);
     return x;
 }

 void Pop(int x){
     int rt=root[x];
     root[x]=merge(t[rt].l,t[rt].r);
     sum[x]--; add[x]-=t[rt].val;
     t[rt].l=t[rt].r=t[rt].val=;
 }

 void dfs(int u){
     add[u]=sum[u]=;
     for (int i=head[u],vv; i; i=e[i].next){
         dfs(vv=e[i].to);
         add[u]+=add[vv]; sum[u]+=sum[vv];
         root[u]=merge(root[u],root[vv]);
         while (add[u]>m) Pop(u);
     }
     add[u]+=v[u]; sum[u]++;
     t[u]=(heap){,,v[u]};
     root[u]=merge(root[u],u);
     while (add[u]>m) Pop(u);
     ans=max(ans,sum[u]*w[u]);
 }

 int main(){
     scanf("%d%d", &n, &m); int s;
     ; i<=n; i++){
         int x;
         scanf("%d%lld%lld", &x, &v[i], &w[i]);
         insert(x,i); if (!x) s=i;
     }
     dfs(s);
     printf("%lld\n", ans);
     ;
 }

bzoj1455: 罗马游戏 + bzoj2809: Dispatching(可并堆)的更多相关文章

  1. [bzoj1455]罗马游戏_左偏树_并查集

    罗马游戏 bzoj-1455 题目大意:给你n个人,2种操作,m次操作:1.将i号士兵所在的集合的最小值删除 2.合并i和j两个士兵所在的团体 注释:$1\le n\le 10^6$,$1\le m ...

  2. [BZOJ1455]罗马游戏 左偏树+并查集

    1455: 罗马游戏 Time Limit: 5 Sec  Memory Limit: 64 MBSubmit: 2285  Solved: 994[Submit][Status][Discuss] ...

  3. BZOJ1455 罗马游戏 左偏树 可并堆

    欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ1455 题意概括 n个人,2种操作. 一种是合并两个人团,一种是杀死某一个人团的最弱的人. 题解 左 ...

  4. Bzoj1455 罗马游戏

    Time Limit: 5 Sec  Memory Limit: 64 MBSubmit: 1622  Solved: 679 Description 罗马皇帝很喜欢玩杀人游戏. 他的军队里面有n个人 ...

  5. 【数据结构】bzoj1455罗马游戏

    Description 罗马皇帝很喜欢玩杀人游戏. 他的军队里面有n个人,每个人都是一个独立的团.最近举行了一次平面几何测试,每个人都得到了一个分数. 皇帝很喜欢平面几何,他对那些得分很低的人嗤之以鼻 ...

  6. BZOJ1455——罗马游戏

    1.题目大意:维护一个数据结构,可以实现合并操作,还能询问最小值 2.分析:这种问题当然是可并堆啦 随便写了一个左偏树QAQ #include <cstdio> #include < ...

  7. [BZOJ1455]罗马游戏(左偏树)

    用并查集和左偏树维护士兵的关系 Code #include <cstdio> #include <algorithm> #define N 1000010 using name ...

  8. BZOJ1455罗马游戏

    左偏树裸题. 题面描述让人意识到了平面几何的重要性. //Achen #include<algorithm> #include<iostream> #include<cs ...

  9. 【BZOJ-1455】罗马游戏 可并堆 (左偏树)

    1455: 罗马游戏 Time Limit: 5 Sec  Memory Limit: 64 MBSubmit: 1355  Solved: 561[Submit][Status][Discuss] ...

随机推荐

  1. cxf webservice 生成wsdl方法参数名称为arg0问题

    在通过cxf生成webservice服务时,如果你是用ServerFactoryBean,那么在生成wsdl时,方法的参数名称会被自动命名为arg0,arg1...,如: <xsd:comple ...

  2. 【Juicer】 一个高效、轻量的前端 (Javascript) 模板引擎

    引用地址:http://juicer.name/docs/docs_zh_cn.html * 一个完整的例子 HTML 代码: <script id="tpl" type=& ...

  3. qt5.5程序打包发布以及依赖【转】

    玩qt5也有一段时间了,惭愧的是一直没有好好的发布过程序,因为写的都是小程序没啥需要用到发布,而且qt也说不上很熟悉,本来打算到基本掌握qt之后再来研究研究怎么打包程序,最近晚上的空闲时间多了,闲着也 ...

  4. php 封装 知识点

    类由众多对象抽象出来的对象由类实例化出来的 成员变量成员方法成员属性 访问修饰符public 公有的protected 受保护的private 私有的 构造函数1.写法特殊2.执行时间特殊 面向对象的 ...

  5. 51nod1265(判断四个点是否共面)

    题目链接:https://www.51nod.com/onlineJudge/questionCode.html#!problemId=1265 题意:中文题诶- 思路:假设现有a, b, c, d四 ...

  6. C++ Tips and Tricks

    整理了下在C++工程代码中遇到的技巧与建议. 0x00 巧用宏定义. 经常看见程序员用 enum 值,打印调试信息的时候又想打印数字对应的字符意思.见过有人写这样的代码 if(today == MON ...

  7. How GitHub Works《Github是如何工作的?》

    https://github.com/blog/920-how-github-works 如果你想知道Github是如何工作的,你可以看查看Zach Holman(@holman)的三篇文章: Hou ...

  8. AVAssetReader+AVAssetReaderTrackOutput播放视频

    该文章引用自:http://www.jianshu.com/p/3d5ccbde0de1 IOS 微信聊天发送小视频的秘密(AVAssetReader+AVAssetReaderTrackOutput ...

  9. android应用程序第一次启动时显示引导界面

    市面上好多优秀的应用(举例新浪微博.UC浏览器)都采用了欢迎页面与使用向导的方式给用户带来了良好的用户体验. 一般来说用户第一次安装应用或者安装了新版本后第一次进入应用都会显示成 欢迎页面-使用向导- ...

  10. MSP430FR4133/4131/4132单片机破解芯片解密多少钱?

    德州仪器MSP430FR4133/4131/4132单片机破解芯片解密多少钱? MSP430FR4133.MSP430FR4131.MSP430FR4132 ####[微信:icpojie]#### ...