bzoj3728: PA2014Final Zarowki

 

Description

有n个房间和n盏灯,你需要在每个房间里放入一盏灯。每盏灯都有一定功率,每间房间都需要不少于一定功率的灯泡才可以完全照亮。
你可以去附近的商店换新灯泡,商店里所有正整数功率的灯泡都有售。但由于背包空间有限,你至多只能换k个灯泡。
你需要找到一个合理的方案使得每个房间都被完全照亮,并在这个前提下使得总功率尽可能小。

Input

第一行两个整数n,k(1<=k<=n<=500000)。
第二行n个整数p[i](1<=p[i]<=10^9),表示你现有的灯泡的功率。
第三行n个整数w[i](1<=w[i]<=10^9),表示照亮每间房间所需要的最小功率。

Output

如果无法照亮每间房间,仅输出NIE。
否则输出最小的总功率。

Sample Input

6 2

12 1 7 5 2 10

1 4 11 4 7 5

Sample Output

33

HINT

解释:将2和10换成4和4。配对方案为1-1,4-4,4-4,5-5,7-7,11-12。

/*题面选最小。。。

    Multiset+贪心可水,原理就是小的灯泡能用则用,不能用的保留,最后再用k次机会滑稽掉剩下的。再有剩余的可以把之前用的灯泡和房间差值变成0来减小。

    这个堆+贪心啊,我承认我傻=-=。想了好久才明白。

    首先对功率和房间从大到小排序,排完以后开始贪心。把符合当前房间条件的所有的灯泡都放进堆里(这个堆也保留之前房间剩余的),在里面选个最小的来提供给房间。但是!当这个堆里啥玩意都没有,即对于这个垃圾房间没有灯泡给它,我们就用一次换灯泡的机会,再给ans加上房间的需求,然后继续即可以。

    然后我就方了,为什么不用改别的灯泡呢=-=,我这个蒟蒻百思不得其解=-=。因为我们使用这次机会以后,首先,对于以后的数据,扩大了可选择的范围(因为灯泡里接下来的最大值被保留了),其次由于不存在可以给这个房间提供的灯泡(因为有的之前给别的了,如果别的被替换,还是得使用一次机会)这次机会必须得用掉,然后对于某个可能被替换掉的灯泡,我相当于把它延后选择,即先把接下来可用灯泡都用掉,最后一定会多剩余一个灯泡(如果有小到没法用的自然剩余,要是较小的在某个房间可用,便会剩下一个大的)。

    综上而言,这个贪心是正确的,因为它没有导致答案变大(使用最小),且保证了可解(保留剩下最大)。*/

 1 #include<cstdio>
 2 #include<queue>
 3 #include<algorithm>
 4 using namespace std;
 5 int n,k;
 6 const int N=500500;
 7 priority_queue<int > red;
 8 priority_queue<int,vector<int>,greater<int> > q;
 9 long long ans;
10 int p[N],w[N];
11 inline bool m_s(int a,int b){
12     return a>b;
13 }
14 int main(){
15     scanf("%d%d",&n,&k);
16     for(int i=1;i<=n;i++)   scanf("%d",p+i);
17     for(int i=1;i<=n;i++)   scanf("%d",w+i);
18     sort(p+1,p+1+n,m_s);sort(w+1,w+n+1,m_s);
19     for(int i=1,j=1;i<=n;i++){
20         while(p[j]>=w[i])   q.push(p[j++]);
21         if(!q.empty()){
22             ans+=q.top();
23             red.push(q.top()-w[i]);q.pop();continue;
24         }
25         k--,ans+=w[i];
26         if(k<0) {printf("NIE\n");return 0;}
27     }
28     while(k--)ans-=red.top(),red.pop();
29     printf("%lld\n",ans);
30 }

 

posted @ 2017-07-26 21:46  Troywar  阅读(364)  评论(0编辑  收藏  举报