[BZOJ2288&BZOJ1150]一类堆+链表+贪心问题
今天我们来介绍一系列比较经典的堆+链表问题.这类问题的特点是用堆选取最优解,并且通过一些加减操作来实现"反悔".
在看题之前,我们先来介绍一个神器:手写堆.
手写堆的一大好处就是可以可以访问,或者删除堆中的某个特定元素.
手写堆其实就是在模拟二叉堆的比较大小过程,比如:
inline void up(int id)
{
if(id==)return;
if(h[id>>]<h[id])
swap(h[id>>],h[id]),up(id>>);
}
inline void down(int id)
{
id<<=;if(id>top)return;
if(id<top&&h[id^]>h[id])id^=;
if(h[id>>]<h[id])
swap(h[id>>],h[id]),down(id);
}
这样我们已经可以查找最值了.如果我们想删除某个值,就直接把它的权值设为无穷大/小,再调用down函数.
如果我们还想支持查询某个标号对应的值(下面两个题都会用到)
可以多加一个match数组记录编号.当然,这里的h既要记录编号又要记录权值.
完整的像这样,当然,读者可以结合自己的风格打出自己的手写堆:
int match[N],top;
struct node
{
int val,pos;
node(int a=,int b=){val=a,pos=b;}
}h[N];
inline void up(int i)
{
while(i>&&h[i].val<h[i>>].val)
match[h[i>>].pos]=i,swap(h[i>>],h[i]),i>>=;
match[h[i].pos]=i;
}
inline void down(int i)
{
int to;
while((i<<)<=top)
{
to=(i<<);if(to<top&&h[to+].val<h[to].val)++to;
if(h[to].val<h[i].val)
match[h[to].pos]=i,swap(h[i],h[to]),i=to;
else break;
}
match[h[i].pos]=i;
}
inline void push(int val,int pos)
{h[++top]=node(val,pos);up(top);}
inline void pop(int i)
{h[i].val=inf;down(i);}
拿到了我们的强力武器,话不多说,我们先看一道题:
1150: [CTSC2007]数据备份Backup
Time Limit: 10 Sec Memory Limit: 162 MB
Description

Input
Output
输出应由一个正整数组成,给出将2K个相异的办公楼连成k对所需的网络电缆的最小总长度。
Sample Input
1
3
4
6
12
Sample Output
然后,如果我们又选到了节点i,这时候再给ans加上它的val,我们实质上选了val[i-1]和val[i+1],

#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
const int N=,inf=0x7fffffff;
int n,m,s[N],d[N],cnt,top;
int match[N];
struct node
{
int val,pos;
node(int a=,int b=){val=a,pos=b;}
}h[N];
inline void up(int i)
{
while(i>&&h[i].val<h[i>>].val)
match[h[i>>].pos]=i,swap(h[i>>],h[i]),i>>=;
match[h[i].pos]=i;
}
inline void down(int i)
{
int to;
while((i<<)<=top)
{
to=(i<<);if(to<top&&h[to+].val<h[to].val)++to;
if(h[to].val<h[i].val)
match[h[to].pos]=i,swap(h[i],h[to]),i=to;
else break;
}
match[h[i].pos]=i;
}
inline void push(int val,int pos)
{h[++top]=node(val,pos);up(top);}
inline void pop(int i)
{h[i].val=inf;down(i);}
int pre[N],nxt[N];
int main()
{
register int i,j,k;
scanf("%d%d",&n,&m);
for(i=;i<=n;++i)scanf("%d",&s[i]);
for(i=;i<n;++i)d[i]=s[i+]-s[i],push(d[i],i);
for(i=;i<n;++i)pre[i]=i-,nxt[i]=i+;
nxt[n-]=;
node x;int l,r,ans=;
for(i=;i<=m;++i)
{
x=h[];ans+=x.val;
l=pre[x.pos],r=nxt[x.pos];
if(!l) pop(),pop(match[r]),pre[nxt[r]]=;
else if(!r) pop(),pop(match[l]),nxt[pre[l]]=;
else
h[].val=h[match[l]].val+h[match[r]].val-h[].val,
pre[x.pos]=pre[l],nxt[pre[l]]=x.pos,
nxt[x.pos]=nxt[r],pre[nxt[r]]=x.pos,
pop(match[l]),pop(match[r]),down();
}
printf("%d\n",ans);
}
下面我们再来看另外一道题,套路是相似的,但是更加有挑战性:
2288: 【POJ Challenge】生日礼物
Time Limit: 10 Sec Memory Limit: 128 MB
Description
ftiasch 18岁生日的时候,lqp18_31给她看了一个神奇的序列 A1, A2, ..., AN. 她被允许选择不超过 M 个连续的部分作为自己的生日礼物。
自然地,ftiasch想要知道选择元素之和的最大值。你能帮助她吗?

Input
第1行,两个整数 N (1 ≤ N ≤ 105) 和 M (0 ≤ M ≤ 105), 序列的长度和可以选择的部分。
第2行, N 个整数 A1, A2, ..., AN (0 ≤ |Ai| ≤ 104), 序列。
Output
一个整数,最大的和。
Sample Input
2 -3 2 -1 2
Sample Output
[BZOJ2288&BZOJ1150]一类堆+链表+贪心问题的更多相关文章
- BZOJ 2288: 【POJ Challenge】生日礼物 堆&&链表
就是堆+链表,十分像 数据备份 对吧? 把相邻的正数和相邻的负数合并成一整个正数块和负数块,最后只剩一些交替相间的正块与负块了吧? 显然,正块的个数<=m时,全部选走就获得了最大权值,否则我们可 ...
- [USACO12FEB]牛券Cow Coupons(堆,贪心)
[USACO12FEB]牛券Cow Coupons(堆,贪心) 题目描述 Farmer John needs new cows! There are N cows for sale (1 <= ...
- [bzoj2288][pojChallenge]生日礼物【贪心+堆+链表】
题目描述 ftiasch 18岁生日的时候,lqp18_31给她看了一个神奇的序列 A1, A2, -, AN. 她被允许选择不超过 M 个连续的部分作为自己的生日礼物. 自然地,ftiasch想要知 ...
- BZOJ_2151_种树_贪心+堆+链表
BZOJ_2151_种树_贪心+堆 Description A城市有一个巨大的圆形广场,为了绿化环境和净化空气,市政府决定沿圆形广场外圈种一圈树.园林部门得到指令后,初步规划出n个种树的位置,顺时针编 ...
- BZOJ1150 [CTSC2007]数据备份Backup 【堆 + 链表】
题目 你在一家 IT 公司为大型写字楼或办公楼(offices)的计算机数据做备份.然而数据备份的工作是枯燥乏味 的,因此你想设计一个系统让不同的办公楼彼此之间互相备份,而你则坐在家中尽享计算机游戏的 ...
- BZOJ1150[CTSC2007]数据备份Backup——模拟费用流+堆+链表
题目描述 你在一家 IT 公司为大型写字楼或办公楼(offices)的计算机数据做备份.然而数据备份的工作是枯燥乏味 的,因此你想设计一个系统让不同的办公楼彼此之间互相备份,而你则坐在家中尽享计算机游 ...
- [luogu3620][APIO/CTSC 2007]数据备份【贪心+堆+链表】
题目描述 你在一家 IT 公司为大型写字楼或办公楼(offices)的计算机数据做备份.然而数据备份的工作是枯燥乏味的,因此你想设计一个系统让不同的办公楼彼此之间互相备份,而你则坐在家中尽享计算机游戏 ...
- BZOJ2151 种树(贪心+堆+链表/wqs二分+动态规划)
dp容易想到,但没法进一步优化了. 考虑贪心,每次选出价值最大的物品.但这显然是不对的因为会影响其他物品的选择. 于是考虑加上反悔操作.每次选出一个物品后,将其相邻两物品删除,再将原物品价值变为相邻两 ...
- BZOJ2288 【POJ Challenge】生日礼物 【堆 + 链表】
题目 ftiasch 18岁生日的时候,lqp18_31给她看了一个神奇的序列 A1, A2, ..., AN. 她被允许选择不超过 M 个连续的部分作为自己的生日礼物. 自然地,ftiasch想要知 ...
随机推荐
- 180803-Spring定时任务高级使用篇
Spring定时任务高级使用篇 前面一篇博文 <Spring之定时任务基本使用篇> 介绍了Spring环境下,定时任务的简单使用姿势,也留了一些问题,这一篇则希望能针对这些问题给个答案 I ...
- vue.js和vue-router和vuex快速上手知识
vue.js和vue-router和vuex快速上手知识 一直以来,认为vue相比react而言,学习成本会更低,会更简单,但最近真正接触后,发现vue的各方面都有做一些客户化的优化,有一些亮点,但也 ...
- javascript实现对html便签等字符的转义
参考链接:https://www.jb51.net/article/152700.htm 请访问以上链接. 本人纯搬迁,防止原作者删除. <script> var HtmlUtil = { ...
- 如何布局您的PC站和移动站,并表达两者之间内容的对应关系
如何布局您的PC站和移动站,并表达两者之间内容的对应关系 目前较流量的PC站与移动站配置方式有三种,百度站在搜索引擎角度将这三种分别称为跳转适配.代码适配和自适应,以下为这三种配置方式的名词解释及 ...
- Netty源码分析第4章(pipeline)---->第4节: 传播inbound事件
Netty源码分析第四章: pipeline 第四节: 传播inbound事件 有关于inbound事件, 在概述中做过简单的介绍, 就是以自己为基准, 流向自己的事件, 比如最常见的channelR ...
- GTK 预置对话框 GtkDialog 文件/颜色/字体选取等 GtkFileSelection
(GTK2) 文档链接 作用:打开一个预置的对话框,如文件选取对话框 GtkFileSelection 效果下图所示 ╰── GtkDialog ├── GtkAboutDialog ├── GtkC ...
- linux go环境安装
方法一 这次将源码包安装的目录是是/root下. 1.官网下载源码包. 官网链接:https://golang.org/dl/ wget https://storage.googleapis.co ...
- can总线实现stm32的IAP
使用stm32f105rct6的can通信做IAP,实现固件的远程更新功能.IAP的实现包括两个程序:BootLoader和应用程序.启动过程先启动BootLoader,等待1s,若接收到烧写指令则开 ...
- Node.js开发入门—套接字(socket)编程
Node.js的net模块提供了socket编程接口,方便我们利用较为底层的套接字接口来实现应用协议.这次我们看一个简单的回显服务器示例,包括服务端和客户端的代码. 代码 分服务器和客户端两部分来说吧 ...
- [奇葩问题] Error Domain=NSURLErrorDomain Code=-1003
问题描述: 新上线的产品,ios同事拿着一串报错来找我,日志如下:err =Error Domain=NSURLErrorDomain Code=-1003 "未能找到使用指定主机名的服务器 ...