种了一下午的树,终于给搞明白了((多谢各位大神的题解)(题解就不能讲清楚点吗(看不见看不见)));

你有k个树,你可以种在一条直线上,每个位置都有一个价值,如果你把树种在这里就可以获得这个价值,但是条件是不能有任意两棵树相邻;

当然你可以种0到k任意棵树;

怎么办呢?这道题也是一个(少见)的良心题;我先写了个无脑20分暴力,枚举这个位置选还是不选;

#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int maxn=5e5+;
typedef long long ll;
int n,k;
int a[maxn];
int b[maxn];
ll ans;
ll cnt;
void dfs(int x)
{
if(x==n+)
{
ll sum=;
int num=;
for(int i=;i<=n;i++)
{
if(b[i]==&&b[i-]==) return ;
if(b[i]==)
{
num++;
if(num>k) return ;
sum+=(ll)a[i];
}
}
ans=max(ans,sum);
return ;
}
b[x]=;dfs(x+);
b[x]=;dfs(x+);
} int main()
{
scanf("%d%d",&n,&k);
for(int i=;i<=n;i++) scanf("%d",&a[i]);
dfs();
printf("%lld",ans);
return ;
}

但是怎么能甘心20分,想到这个是一个直线,可能用dp,随便糊了一个就过了50分;

就是f[i][j]表示当前从1到i位置一共种了j棵树最大价值;

如果当前不种树,直接复制f[i-1][j],要是想种树,就复制f[i-2][j-1](因为不能相邻)

#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int maxn=5e5+;
typedef long long ll;
int n,k;
ll a[maxn];
ll ans;
ll f[][];
int main()
{
scanf("%d%d",&n,&k);
for(int i=;i<=n;i++) scanf("%lld",&a[i]);
f[][]=a[];
for(int i=;i<=n;i++)
{
for(int j=;j<=k;j++)
{
f[i][j]=max(f[i-][j],f[i-][j-]+a[i]);
ans=max(ans,f[i][j]);
}
}
printf("%lld",ans);
return ;
}

空间时间限制,我不能拿满分,好气哦;

终于,我打开了题解(眼睛放出光芒)

three months later……

我们不用管什么谁先种谁后种,只要最大值;

我们可以用优先队列,这个是怎么想出来的呢?面对连续的3个数(首位或末尾位置随意),标号1,2,3;

如果我们选2号坑,那么1,3就废了;

如果选1.3,那么2就不能选了;(我表示不想分析这种事情)(优先队列帮您实现!!)

我们默认堆顶就是我们要找的最大值,但是我们需要经过一些处理;

将每个点的左右端位置记录下来l[i]=i-1;r[i]=i+1;

先将n个坑扔进去,那么堆顶将是价值最大的那个,我们取出来ans+=val(为什么,1,3怎么办?)

没关系,因为我们选1就不能选1,3,那我们就不选1,3了,将2的值取出后再更新2的值,

val[2]=val[1]+val[3]-val[2];(ans需要更新的值)

但是我们要避免重复选取,将vis[1]=vis[3]=1;(不在特殊访问)

还有就是这样做相当于分成了三个一组三个一组,我们将2的值更新后要将左右位置更新(相当于变成了0,2,4)

这样我们其实解决了三个一组首尾会相连的问题(如果我们要选1,3,我们会再次更新2.此时是0,2,4,模式,他们就不会相连;)

(在选1,3时已经和0,4比较了)

处理完成;

当堆里都是负数的时候就直接退出就行了;

#include<cstdio>
#include<cstring>
#include<queue>
#include<algorithm>
using namespace std;
const int maxn=5e5+;
typedef long long ll;
int a[maxn];
int n,k;
int l[maxn],r[maxn];
priority_queue<pair<int,int> > q;
ll ans;
int vis[maxn];
int main()
{
scanf("%d%d",&n,&k);
for(int i=;i<=n;i++)
{
scanf("%d",&a[i]);
//a[i].val=-a[i].val;
q.push(make_pair(a[i],i));
l[i]=i-;
r[i]=i+;
}
r[]=;l[n+]=n;
while(k--)
{
while(vis[q.top().second]) q.pop();
int x=q.top().first;
int i=q.top().second;
q.pop();
if(x<) break;
ans+=(ll)x;
a[i]=a[l[i]]+a[r[i]]-a[i];
vis[l[i]]=;vis[r[i]]=;
l[i]=l[l[i]];r[l[i]]=i;
r[i]=r[r[i]];l[r[i]]=i;
q.push(make_pair(a[i],i));
}
printf("%lld",ans);
return ;
}

看这些做法和代码长度,你觉得这题(它难吗)(逃~);

P1484 种树——数据结构优先队列的更多相关文章

  1. 洛谷 P1484 种树(优先队列,贪心,链表)

    传送门 解题思路 第一眼的贪心策略:每次都选最大的. 但是——不正确! 因为选了第i个树,第i-1和i-1棵树就不能选了.所以,要有一个反悔操作. 选了第i个后,我们就把a[i]的值更新为a[l[i] ...

  2. 洛谷P1484 种树&洛谷P3620 [APIO/CTSC 2007]数据备份 题解(堆+贪心)

    洛谷P1484 种树&洛谷P3620 [APIO/CTSC 2007]数据备份 题解(堆+贪心) 标签:题解 阅读体验:https://zybuluo.com/Junlier/note/132 ...

  3. P1484 种树

    P1484 种树 题意: 在n个数中选出至多k个数,且两两不相邻,并使所选数的和最大. n<=500000  思路 先建一个堆,把所有点扔进去,当取出队首元素时累加到答案时,把它和它左右两个点一 ...

  4. JAVA数据结构--优先队列(堆实现)

    优先队列(堆)的定义 堆(英语:Heap)是计算机科学中一类特殊的数据结构的统称.堆通常是一个可以被看做一棵树的数组对象.在队列中,调度程序反复提取队列中第一个作业并运行,因为实际情况中某些时间较短的 ...

  5. javascript数据结构-优先队列

    这里之所以扩充一个 有限队列 是因为,生活使用中队列通常会附加优先级,比如排队买票,一般老人和军人等会有优先权限. 实现:继承上篇的 普通队列实现.这里用一种方法,入队的时候,进行排序插入到指定位置, ...

  6. 洛谷 P1484 种树

    题目描述 cyrcyr今天在种树,他在一条直线上挖了n个坑.这n个坑都可以种树,但为了保证每一棵树都有充足的养料,cyrcyr不会在相邻的两个坑中种树.而且由于cyrcyr的树种不够,他至多会种k棵树 ...

  7. [洛谷P1484] 种树

    题目类型:堆+贪心 传送门:>Here< 题意:有\(N\)个坑,每个坑可以种树,且获利\(a[i]\)(可以为负).任何相邻两个坑里不能都种树,问在最多种\(K\)棵树的前提下的最大获利 ...

  8. 【洛谷】【堆+贪心】P1484 种树

    [题目描述:] cyrcyr今天在种树,他在一条直线上挖了n个坑.这n个坑都可以种树,但为了保证每一棵树都有充足的养料,cyrcyr不会在相邻的两个坑中种树.而且由于cyrcyr的树种不够,他至多会种 ...

  9. 题解 P1484 种树

    题目 传送门 cyrcyr今天在种树,他在一条直线上挖了 n 个坑. 这n个坑都可以种树,但为了保证每一棵树都有充足的养料,cyrcyr不会在相邻的两个坑中种树. 而且由于cyrcyr的树种不够,他至 ...

随机推荐

  1. access 数据库创建表SQL语法

    create table R_CAIFA_B13 ( ID AUTOINCREMENT PRIMARY KEY, XB varchar(255), C1 varchar(50), C2 varchar ...

  2. vue-filters(过滤器)

    局部过滤器: <html> <head> <title>vue</title> <meta charset="utf-8"&g ...

  3. 定位之z-index

    我们已经知道固定定位(fixed)和绝对定位(absolute)可以让盒子浮起来 相对定位(relactive)虽然不能让盒子浮起来,但也是可以让图层浮起来 那么既然大家都可以浮起来,就会存在一个问题 ...

  4. form表单提交后结果乱码的解决方法

    1.产生乱码原因:表单提交使用的method="get",get方式数据都是通过地址栏传输,数据会以iso-8859-1方式传输,因此产生乱码 2.概念:URI: Uniform ...

  5. 使用帅气的cordic算法进行坐标系互转及log10的求解

    参考博客 https://blog.csdn.net/u010712012/article/details/77755567 https://blog.csdn.net/Reborn_Lee/arti ...

  6. JAVA中对象的克隆及深拷贝和浅拷贝

    使用场景: 在日常的编程过程 中,经常会遇到,有一个对象OA,在某一时间点OA中已经包含了一些有效值 ,此时可能会需一个和OA完全相对的新对象OB,并且要在后面的操作中对OB的任何改动都不会影响到OA ...

  7. Introduction of Machine Learning

    李宏毅主页 台湾大学语音处理实验室 人工智慧.机器学习与深度学习间有什么区别? 人工智能——目标 机器学习——手段 深度学习——机器学习的一种方法 人类设定好的天生本能 Machine Learnin ...

  8. ARP、Proxy ARP、Gratuitous ARP

    Proxy ARP 什么是Proxy ARP? 一个主机A(通常是路由器)有意应答另一个主机B的ARP请求(ARP requests).主机A通过伪装其身份,承担起将分组路由到真实目的地的责任.代理A ...

  9. python3 虚拟环境

    一.python中的虚拟环境 1.虚拟环境:局部的,独立的python环境,完全模拟系统全局python环境的使用 二.安装 http://virtualenv.pypa.io/en/latest/u ...

  10. sklearn逻辑回归实战

    目录 题目要求 ex2data1.txt处理 方案一:无多项式特征 方案二:引入多项式特征 ex2data2.txt处理 两份数据 ex2data1.txt ex2data2.txt 题目要求 根据学 ...