【BZOJ】2151 种树

【算法】贪心+堆

【题意】n个数字的序列,要求选择互不相邻的k个数字使和最大。

【题解】

贪心,就是按一定顺序选取即可最优,不会反悔。

考虑第一个数字选择权值最大的,那么它相邻的两个数字就不能选择,那么我们可以把这三个数字视为一个整体。操作为将pre[pre[x]]和x和suc[suc[x]]连接起来。

由于可能依然有选择相邻两个数字的可能性,将中间的权值置为A[pre[x]]+A[suc[x]]-A[x],这个权值记录在中间,但实际上代表相邻三个数字的新权值。

若再次选择这个区间,那么就是把区间更新到周围五个数字了(这里的数字有可能已经是一段区间),如此可以不断扩大。

贪心原理:若没有间隔种的限制,每次都取大就是最优的。那么多了这个限制之后,取大依然优但却会影响旁边两个数字的选择。

为了使我们依然能贪心,就设置一个反悔的机会,即设置一个新权值。那么如果非要选择旁边两个就相当于选择中间的权值两次,每次把影响区间扩大2并视之为一个数字。

因为贪心就不会反悔的性质,既然会选择这个决策就一定不会悔改。这个反悔的机会实质上是扩大你选择数字的影响范围,一旦扩大就一定不会反悔,因为一定是最优的。

每次出现新权值用堆维护。记得开始先特判无法种k棵的情况,因为后面很难判断。

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<queue>
using namespace std;
const int maxn=200010;
struct cyc{
    int num,ord;
    bool operator < (const cyc &x)const
    {return num<x.num;}
}qp;
priority_queue<cyc>q;

int a[maxn],pre[maxn],suc[maxn],n,k;
bool f[maxn];
void del(int x)
{
    f[x]=1;
    suc[pre[x]]=suc[x];
    pre[suc[x]]=pre[x];
    suc[x]=pre[x]=0;
}
int main()
{
    scanf("%d%d",&n,&k);
    for(int i=1;i<=n;i++)
    {
        scanf("%d",&a[i]);
        q.push((cyc){a[i],i});
        pre[i]=i-1;suc[i]=i+1;
    }
    pre[1]=n;suc[n]=1;
    int ans=0;
    if(k>n/2){printf("Error!");return 0;}
    for(int i=1;i<=k;i++)
    {
        while(f[q.top().ord])q.pop();
        qp=q.top();q.pop();
        ans+=qp.num;
        int A=pre[qp.ord],B=suc[qp.ord];
        a[qp.ord]=a[A]+a[B]-qp.num;
        del(pre[qp.ord]);del(suc[qp.ord]);
        q.push((cyc){a[qp.ord],qp.ord});
    }
    printf("%d",ans);    
    return 0;
}
View Code
原文地址:https://www.cnblogs.com/onioncyc/p/7089493.html