首页 > 其他 > 详细

BZOJ 1150: [CTSC2007]数据备份Backup 优先队列+贪心+链表

时间:2019-02-10 21:06:12      阅读:194      评论:0      收藏:0      [点我收藏+]

题解:我们先得到两个楼之间的距离,D[i]表示第i栋楼和第i+1栋楼之间的距离,我们要选出最小的k个数,然后就有两种情况

1.选择了D[i],那么D[i-1]和D[i+1]都不能选择了
2.选择了D[i+1]和D[i-1],然后无法选择D[i].

既要么选D[i]不选D[i-1]和D[i+1],要么选D[i-1]和D[i+1]不选D[i];(怎么证明的也搞不太清楚)我们用优先队列来得到每次最小的距离,

我们发现在选取一个数之后,只对左右两边的数有影响,选了D[i]后,我们把D[i-1]+D[i+1]-D[i]加入队列,同时D[i-1]和D[i+1]不能再选了,我们就把D[i]旁边的边删除,然后再打上标记

和双链表中的删除一个节点类似,这里用数组代替了链表。

还有由于我的优先队列里放的是整个结构体,所以每次取出堆顶后,L和R一定要使用数组中的,而不能直接使用当前取出的节点的L和R(这个地方wa了好几次),因为有些楼的左右节点变了,而优先队列中的还没变。

#include<bits/stdc++.h>
using namespace std;
const int inf = 0x3f3f3f3f;
const int maxn = 2e5 + 5;
//数组代替链表 每个节点代表一栋楼的信息,以及左边,右边的楼的编号
struct node{  
    int pos, dis, Lpos, Rpos; 
    node(int pos = 0, int dis = 0, int Lpos = 0, int Rpos = 0){ pos = pos, dis = dis, Lpos = Lpos, Rpos = Rpos; }
    bool operator<(const node &other)const{
        return dis > other.dis; 
    }
}a[maxn];
bool vis[maxn];
int b[maxn],n,k;
priority_queue<node> que;
long long ans;
void solve()
{
    while (k--)
    {
        while (1)
        {
            node cur = que.top();
            que.pop();
            if (vis[cur.pos]) continue;
            int pos = cur.pos, Lpos = a[pos].Lpos, Rpos = a[pos].Rpos;    /*优先队列中放整个结构体的一定要注意这个地方,这里Lpos和Rpos一定要用
            数组中的L,R,而不能直接使用cur.Lpos,cur.Rpos 我在这里卡了好长时间*/
            vis[Lpos] = vis[Rpos] = true; //这个楼已使用,那么它旁边的楼就不能再使用,打上标记,pos这个位置还会继续使用,因此pos不打标记
            ans += 1LL*cur.dis;
            a[pos].dis = a[Lpos].dis + a[Rpos].dis - a[pos].dis;// pos位置继续使用,但改变距离
            a[pos].Lpos = a[Lpos].Lpos; a[pos].Rpos = a[Rpos].Rpos;
            a[a[Rpos].Rpos].Lpos = pos; a[a[Lpos].Lpos].Rpos = pos; //改变左右相邻的楼
            que.push(a[pos]);
            break;
        }

    }
}

int main()
{

    cin >> n >> k;
    for (int i = 1; i <= n; i++)
    {
        cin >> b[i];
    }
    for (int i = 1; i < n; i++)
    {
        a[i].dis = b[i + 1] - b[i];
        a[i].pos = i;
        a[i].Lpos = i - 1;
        a[i].Rpos = i + 1;
        que.push(a[i]);
    }
    a[n - 1].Rpos = 0;
    a[0].dis = a[n].dis = inf;
    solve();
    cout << ans;
    return 0;
}

再来一个不放结构体的

struct cmp{
    bool operator()(int a, int b)
    {
        return D[a] > D[b];
    }
};
priority_queue<int, vector<int>, cmp>;

 

BZOJ 1150: [CTSC2007]数据备份Backup 优先队列+贪心+链表

原文:https://www.cnblogs.com/xiaoguapi/p/10360143.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!