BZOJ2151 种树(贪心+堆+链表/wqs二分+动态规划)
dp容易想到,但没法进一步优化了。
考虑贪心,每次选出价值最大的物品。但这显然是不对的因为会影响其他物品的选择。
于是考虑加上反悔操作。每次选出一个物品后,将其相邻两物品删除,再将原物品价值变为相邻两物品价值和-原物品价值。这样如果再次选择该物品就可以达到改为选择相邻两物品的效果。并且最优方案中相邻两物品一定要么都选要么都不选,否则不如选择原物品。
这种带反悔的贪心策略似乎类似地在网络流算法中出现,应该是一个比较普遍的做法,然而并不会证。
#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstdlib>
#include<cstring>
#include<algorithm>
#include<queue>
using namespace std;
int read()
{
int x=,f=;char c=getchar();
while (c<''||c>'') {if (c=='-') f=-;c=getchar();}
while (c>=''&&c<='') x=(x<<)+(x<<)+(c^),c=getchar();
return x*f;
}
#define N 200010
int n,m,a[N],pre[N],nxt[N],ans;
bool flag[N];
struct data
{
int i,x;
bool operator <(const data&a) const
{
return x<a.x;
}
};
priority_queue<data> q;
int main()
{
#ifndef ONLINE_JUDGE
freopen("bzoj2151.in","r",stdin);
freopen("bzoj2151.out","w",stdout);
const char LL[]="%I64d\n";
#else
const char LL[]="%lld\n";
#endif
n=read(),m=read();
if (m>n/) {cout<<"Error!";return ;}
for (int i=;i<=n;i++) a[i]=read(),q.push((data){i,a[i]});
for (int i=;i<n;i++) nxt[i]=i+;
for (int i=;i<=n;i++) pre[i]=i-;
pre[]=n,nxt[n]=;
for (int i=;i<=m;i++)
{
while (flag[q.top().i]) q.pop();
data t=q.top();q.pop();
ans+=t.x;
int x=pre[t.i],y=nxt[t.i];
a[t.i]=a[x]+a[y]-a[t.i];
q.push((data){t.i,a[t.i]});
flag[x]=flag[y]=;
nxt[pre[x]]=t.i,pre[t.i]=pre[x];
pre[nxt[y]]=t.i,nxt[t.i]=nxt[y];
}
cout<<ans;
return ;
}
不过dp真的没救了吗?看到这个恰好选k个非常容易让人联想到wqs二分。于是二分多选一个数的代价,跑正常的dp,就过掉了。
#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstdlib>
#include<cstring>
#include<algorithm>
#include<queue>
using namespace std;
int read()
{
int x=,f=;char c=getchar();
while (c<''||c>'') {if (c=='-') f=-;c=getchar();}
while (c>=''&&c<='') x=(x<<)+(x<<)+(c^),c=getchar();
return x*f;
}
#define N 200010
#define inf 2000000010
int n,m,a[N],f[N][][],g[N][][],ans;
void chkmax(int &x,int &y,int a,int b)
{
if (a>x||a==x&&b<y) x=a,y=b;
}
bool check(int k)
{
memset(f,,sizeof(f)),memset(g,,sizeof(g));
f[][][]=a[]+k,g[][][]=;
f[][][]=f[][][]=-inf;
for (int i=;i<=n;i++)
{
f[i][][]=f[i-][][]+a[i]+k,g[i][][]=g[i-][][]+;
if (i<n) f[i][][]=f[i-][][]+a[i]+k,g[i][][]=g[i-][][]+;
if (f[i-][][]>f[i-][][]||f[i-][][]==f[i-][][]&&g[i-][][]<g[i-][][]) f[i][][]=f[i-][][],g[i][][]=g[i-][][];
else f[i][][]=f[i-][][],g[i][][]=g[i-][][];
if (f[i-][][]>f[i-][][]||f[i-][][]==f[i-][][]&&g[i-][][]<g[i-][][]) f[i][][]=f[i-][][],g[i][][]=g[i-][][];
else f[i][][]=f[i-][][],g[i][][]=g[i-][][];
}
int x=f[n][][],y=g[n][][];
chkmax(x,y,f[n][][],g[n][][]);
chkmax(x,y,f[n][][],g[n][][]);
chkmax(x,y,f[n][][],g[n][][]);
if (y<=m) {ans=x-m*k;return ;}
else return ;
}
int main()
{
#ifndef ONLINE_JUDGE
freopen("force.in","r",stdin);
freopen("force.out","w",stdout);
const char LL[]="%I64d\n";
#else
const char LL[]="%lld\n";
#endif
n=read(),m=read();
if (m>n/) {cout<<"Error!";return ;}
for (int i=;i<=n;i++) a[i]=read();
int l=-,r=;
while (l<=r)
{
int mid=(l+r)/;
if (check(mid)) l=mid+;
else r=mid-;
}
cout<<ans;
return ;
}
BZOJ2151 种树(贪心+堆+链表/wqs二分+动态规划)的更多相关文章
- BZOJ_2151_种树_贪心+堆+链表
BZOJ_2151_种树_贪心+堆 Description A城市有一个巨大的圆形广场,为了绿化环境和净化空气,市政府决定沿圆形广场外圈种一圈树.园林部门得到指令后,初步规划出n个种树的位置,顺时针编 ...
- bzoj2151 种树 双向链表+堆
2151: 种树 Time Limit: 10 Sec Memory Limit: 259 MBSubmit: 1151 Solved: 613[Submit][Status][Discuss] ...
- [bzoj2288][pojChallenge]生日礼物【贪心+堆+链表】
题目描述 ftiasch 18岁生日的时候,lqp18_31给她看了一个神奇的序列 A1, A2, -, AN. 她被允许选择不超过 M 个连续的部分作为自己的生日礼物. 自然地,ftiasch想要知 ...
- [luogu3620][APIO/CTSC 2007]数据备份【贪心+堆+链表】
题目描述 你在一家 IT 公司为大型写字楼或办公楼(offices)的计算机数据做备份.然而数据备份的工作是枯燥乏味的,因此你想设计一个系统让不同的办公楼彼此之间互相备份,而你则坐在家中尽享计算机游戏 ...
- [BZOJ2151] 种树 贪心
2151: 种树 Time Limit: 10 Sec Memory Limit: 259 MBSubmit: 1151 Solved: 613[Submit][Status][Discuss] ...
- BZOJ 2288: 【POJ Challenge】生日礼物 贪心 + 堆 + 链表
好像是模拟费用流 Code: #include <bits/stdc++.h> #define setIO(s) freopen(s".in","r" ...
- [BZOJ2288&BZOJ1150]一类堆+链表+贪心问题
今天我们来介绍一系列比较经典的堆+链表问题.这类问题的特点是用堆选取最优解,并且通过一些加减操作来实现"反悔". 在看题之前,我们先来介绍一个神器:手写堆. 手写堆的一大好处就是可 ...
- BZOJ5252 八省联考2018林克卡特树(动态规划+wqs二分)
假设已经linkcut完了树,答案显然是树的直径.那么考虑这条直径在原树中是怎样的.容易想到其是由原树中恰好k+1条点不相交的链(包括单个点)拼接而成的.因为这样的链显然可以通过linkcut拼接起来 ...
- WQS二分题集
WQS二分,一种优化一类特殊DP的方法. 很多最优化问题都是形如“一堆物品,取与不取之间有限制.现在规定只取k个,最大/小化总收益”. 这类问题最自然的想法是:设f[i][j]表示前i个取j个的最大收 ...
随机推荐
- C#调用c++类的导出函数
C# 需要调用C++东西,但是有不想做成COM,就只好先导出类中的函数处理. 不能直接调用,需单独导出函数 参考:http://blog.csdn.net/cartzhang/article/deta ...
- Hive窗口函数之LAG、LEAD、FIRST_VALUE、LAST_VALUE的用法
一.创建表: create table windows_ss ( polno string, eff_date string, userno string ) ROW FORMAT DELIMITED ...
- Redis系列四 Redis常见配置
redis.conf常见配置 参数说明redis.conf 配置项说明如下:1. Redis默认不是以守护进程的方式运行,可以通过该配置项修改,使用yes启用守护进程 daemonize no2. ...
- kaggle入门--泰坦尼克号之灾(手把手教你)
作者:炼己者 具体操作请看这里-- https://www.jianshu.com/p/e79a8c41cb1a 大家也可以看PDF版,用jupyter notebook写的,视觉效果上感觉会更棒 链 ...
- 一个体验好的Windows 任务栏缩略图开发心得
本文来自网易云社区 作者:孙有军 前言: 对于一个追求极致体验的软件来说,利用好系统的每一点优秀的特性,将会大大提高软件的品质. Windows vista以来任务栏缩略图,及Win + TAB的程序 ...
- Net Core学习笔记
Net Core 官网:https://dotnet.github.io/ Net Core Api: https://docs.microsoft.com/en-us/dotnet/api/?vie ...
- nginx支持php配置
location / { root /wwwroot/phptest; index index.html index.htm index.php; } location ~ \.(php|php5)$ ...
- java 泛型历史遗留问题
Map<String,Integer> hashMap = new HashMap<String,Integer>(); hashMap.put(); // hashMap.p ...
- 软件测试的基础-摘自《selenium实践-基于电子商务平台》
软件测试的方法 一.等价类划分法 等价类划分法是把所有可能的输入数据,即程序的输入域划分成若干部分(子集),然后从每一个子集中选取少量具有代表性的数据作为测试用例. 有两种不同的情况:有效等价和无效等 ...
- 一个简单的页面弹窗插件 jquery.pageMsgFrame.js
页面弹窗是网站中常用的交互效果,它可以强提示网站的某些信息给用户,或者作用于某些信息的修改等等功能. 这几天在做一个项目的时候,就顺捎把这个插件写一下,栽棵树,自己乘凉吧. 原创博文,转载请注明出处: ...