洛谷 P1484 种树(优先队列,贪心,链表)
传送门
解题思路
第一眼的贪心策略:每次都选最大的。
但是——不正确!
因为选了第i个树,第i-1和i-1棵树就不能选了。所以,要有一个反悔操作。
选了第i个后,我们就把a[i]的值更新为a[l[i]]+a[r[i]]-a[i]。
然后这样如果发现选i-1和i+1更优时,再次加上a[i],结果就变成了a[i]+a[l[i]]+a[r[i]]-a[i]=a[l[i]]+a[r[i]]。
然后这时再更新l[i]和r[i],把左边和右边两个节点删去。
因为每一次会比上一次多种一棵,所以循环k次,求一个ans即为答案。
也许会有疑问,当i-1或i+1是负数时,不选这个负数明显更优,但是a[i-1]+a[i+1]是包含了这个负数的。
其实我们考虑,当有一个是负数,且a[i-1]+a[i+1]>a[i]时,很显然是另一个正数大于a[i],而这个正数一定会比i这棵树先种,所以不必考虑这种情况。
最后,当现在操作的最大利益已经是负数或零了,可以直接break掉,因为树是最大k棵。
AC代码
#include<iostream>
#include<cstdio>
#include<queue>
using namespace std;
const int maxn=;
int n,k,l[maxn],r[maxn],a[maxn],vis[maxn];
long long ans;
struct node{
int id,value;
bool operator < (const node &x)const{
return value<x.value;
}
node(int a,int b):id(a),value(b){}
};
priority_queue<node> q;
int main(){
cin>>n>>k;
for(int i=;i<=n;i++){
scanf("%d",&a[i]);
l[i]=i-;
r[i]=i+;
q.push(node(i,a[i]));
}
r[]=;
l[n+]=n;
while(k--){
while(vis[q.top().id]){
q.pop();
}
node x=q.top();
q.pop();
if(x.value<=) break;
ans+=x.value;
int id=x.id;
vis[l[id]]=vis[r[id]]=;
a[id]=a[l[id]]+a[r[id]]-a[id];
x.value=a[id];
r[l[l[id]]]=id;
l[r[r[id]]]=id;
l[id]=l[l[id]];
r[id]=r[r[id]];
q.push(x);
}
cout<<ans;
return ;
}
洛谷 P1484 种树(优先队列,贪心,链表)的更多相关文章
- 洛谷P1484 种树&洛谷P3620 [APIO/CTSC 2007]数据备份 题解(堆+贪心)
		
洛谷P1484 种树&洛谷P3620 [APIO/CTSC 2007]数据备份 题解(堆+贪心) 标签:题解 阅读体验:https://zybuluo.com/Junlier/note/132 ...
 - [洛谷P1484] 种树
		
题目类型:堆+贪心 传送门:>Here< 题意:有\(N\)个坑,每个坑可以种树,且获利\(a[i]\)(可以为负).任何相邻两个坑里不能都种树,问在最多种\(K\)棵树的前提下的最大获利 ...
 - 洛谷 P1484 种树
		
题目描述 cyrcyr今天在种树,他在一条直线上挖了n个坑.这n个坑都可以种树,但为了保证每一棵树都有充足的养料,cyrcyr不会在相邻的两个坑中种树.而且由于cyrcyr的树种不够,他至多会种k棵树 ...
 - 洛谷P1250种树(贪心)
		
题目描述 一条街的一边有几座房子.因为环保原因居民想要在路边种些树.路边的地区被分割成块,并被编号成1..N.每个部分为一个单位尺寸大小并最多可种一棵树.每个居民想在门前种些树并指定了三个号码B,E, ...
 - 洛谷 P1484 种树 题解
		
题面 这是一道标准的带反悔贪心: 利用大根堆来维护最大值: 当选择了num[i]后,反悔了,反之选择选了num[i-1]和num[i+1]时获利便增加了num[i-1]+num[i+1]-num[i] ...
 - Guard Duty (medium) Codeforces - 958E2 || (bzoj 2151||洛谷P1792) 种树 || 编译优化
		
https://codeforces.com/contest/958/problem/E2 首先求出N个时刻的N-1个间隔长度,问题就相当于在这些间隔中选K个数,相邻两个不能同时选,要求和最小 方法1 ...
 - 【洛谷】【堆+贪心】P1484 种树
		
[题目描述:] cyrcyr今天在种树,他在一条直线上挖了n个坑.这n个坑都可以种树,但为了保证每一棵树都有充足的养料,cyrcyr不会在相邻的两个坑中种树.而且由于cyrcyr的树种不够,他至多会种 ...
 - 洛谷P1996 约瑟夫问题【链表】
		
题目:https://www.luogu.org/problemnew/show/P1996 题意: 约瑟夫环.每次取出第m个,第2m个...... 思路: 链表维护.[感觉很少有用到链表.]非常经典 ...
 - 洛谷P4064 [JXOI2017]加法(贪心 差分)
		
题意 题目链接 Sol 这题就是一个很显然的贪心... 首先二分一个答案,然后check是否可行.check的时候我们需要对每个位置\(i\),维护出所有左端点在\(i\)左侧,右端点在\(i\)右侧 ...
 
随机推荐
- Python使用XML操作mapnik,实现复杂标注(Multi line text symbolizer)
			
test.py import mapnik stylesheet = 'world_style.xml' image = 'world_style.png' m = mapnik.Map(1200, ...
 - IDC装机检查思路
			
交换机网口 网线 配线架 服务器网口灯 系统网卡驱动
 - JSON.parse 测试
			
第一种 报错 var t = JSON.parse(""); console.log(t); 第二种 正常 var t = JSON.parse('{"AA": ...
 - java总结2
			
1,对象数组,必须指定了数组长度,长度是固定的 2,除了ArrayList<E>以外,类赋值给变量,只有string类拿到的是值,其他类拿到的都是类的地址值, ArrayList<E ...
 - Python字典实现
			
这篇文章描述了在Python中字典是如何实现的. 字典通过键(key)来索引,它可以被看做是关联数组.我们在一个字典中添加3个键/值对: >>> d = {'a': 1, 'b': ...
 - 一个时序图描述从用户在浏览器地址栏输入url并按回车,到浏览器显示相关内容的各个过程
			
其实考察的就是一次HTTP请求所经过的过程和Spring或者SpringMVC怎么调用dispatcherServlet的过程
 - 9.并发编程--ThreadLocal
			
并发编程--ThreadLocal 1. ThreadLocal : * 线程局部变量,是一种多个线程间并发访问变量的解决方案. * 与其使用synchronized等加锁的方式,ThreadLoca ...
 - gdb break 断点设置
			
http://sourceware.org/gdb/current/onlinedocs/gdb/ 断点设置 gdb断点分类: 以设置断点的命令分类: breakpoint 可以根据行号.函数.条件生 ...
 - 图的普里姆(Prim)算法求最小生成树
			
关于图的最小生成树算法------普里姆算法 首先我们先初始化一张图: 设置两个数据结构来分别代表我们需要存储的数据: lowcost[i]:表示以i为终点的边的最小权值,当lowcost[i]=0说 ...
 - IO负载高来源定位pt-ioprofile
			
1.使用top -d 1 查看%wa是否有等待IO完成的cpu时间,简单理解就是指cpu等待磁盘写入完成的时间:IO等待所占用的cpu时间的百分比,高过30%时IO压力高: 2.使用iostat -d ...