【codevs 2382】挂缀

题目描述 Description
“珠缀花蕊,人间几多酸泪”……
挂缀在很早就被人们作为一种装饰品,垂坠的风韵,华丽摇曳的摆动,展现出一种与众不同的优雅与高贵。而我们的主人公小Q,正想买一条漂亮的挂缀放在寝室里作为装饰。
挂坠的构成,是由若干粒缀珠相互连接而成。每一个缀珠由三部分组成:分别是珠子、珠子上方的连接环与珠子下方的挂钩(如下图) 。我们可以简单的认为从上往下数的第 i 个缀珠是将它的连接环套在其上方(也就是第 i-1 个)缀珠的挂钩之上(第一个除外) 。小 Q想买一根足够长的挂缀,这样显得更有韵味☺

然而商店的老板告诉小Q,挂缀是不可能做到任意长的,因为每一个珠子都受到重力作用,对其上方的挂钩有一定的拉力,而挂钩的承受能力是有限的。老板还告诉小 Q,他一共拥有 N 个珠缀(假设每一个珠缀都很漂亮,小 Q 都很喜欢) ,每个珠缀都有其各自的重量与承受能力。一个挂缀是稳定的,当且仅当对于其上的每一个珠缀,它下方所有珠缀的重量和(不包含自身)不超过其挂钩的承受能力。
小Q希望她的挂缀尽量长,你能帮她计算出最长可能的稳定挂缀么?当然,如果有多个可选方案,小Q希望总重量最小的。

输入描述 Input Description
第一行包含一个正整数 N,表示商店拥有的珠缀数目。
接下来 N行,每行两个整数(Ci , Wi),分别表示第i 个珠缀的承受能力与重量。

输出描述 Output Description
包行两行。第一行包含一个整数L,表示可以找到的最长稳定挂缀长度。

第二行包含一个整数 W,表示可以找到的长度为 L 的稳定挂缀中的最小重量和。

样例输入 Sample Input
4
3 5
5 1
3 2
4 6

样例输出 Sample Output
3
8

数据范围及提示 Data Size & Hint
对于 30%的数据,N ≤ 10000;
对于 100%的数据,N ≤ 200000;
对于所有的数据,Wi, Ci不超过2^31。

注释里是之前的错误思路
正确思路还是贪心。
只不过用了一个比较奇怪,,比较精妙的策略。
把每个珠子的承受能力和重量加起来,从小到大排序。
按照重量建立一个大根堆。加下一个珠子时,如果发现这个珠子的承受能力太小,不能加到最上面,先不急着忽略,把它跟堆顶元素(重量最大)比较。如果重量小于堆顶元素,则替换。(重量小,总和大,说明承受能力大,更优)。

/*
N ≤ 200000
Wi, Ci不超过2^31 开longlong
下面的一定是承受能力小的 
按照承受能力从小到大排序
按照重量建立一个大根堆
先把承受能力小的放到堆里,意味着把承受能力小的放到挂缀的最下面
统计前缀和(注意重量和不包含自身) 
每次放的时候,比较前缀和跟承受能力。如果前缀和 > 承受能力,说明当前这个不能加到挂缀的最上面
然后再把当前这个跟堆顶(重量最大的)比较
如果当前重量小于堆顶,替换
如果前缀和 < = 承受能力,将当前加到挂缀最上面 
输入n
承受能力  重量
*/
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<queue>
using namespace std;
const int maxn=500000+5;
long long n,sum,l;
struct node
{
    long long f,w,s;
}a[maxn];
bool cmp(node a,node b)
{
    return a.s<b.s;
}
priority_queue<long long>q;
int main()
{
    scanf("%d",&n);
    for(int i=1;i<=n;i++)
    {
        scanf("%d%d",&a[i].f,&a[i].w);
        a[i].s=a[i].f+a[i].w;
    }
    sort(a+1,a+n+1,cmp);
    q.push(a[1].w);
    sum+=a[1].w;
    l++;
    for(int i=2;i<=n;i++)
    {
        if(a[i].f>=sum)
        {
            q.push(a[i].w);
            sum+=a[i].w;
            l++;
        }
        else if(a[i].f<sum)
        {
            long long t=q.top();
            if(t>=a[i].w)
            {
                q.pop();
                q.push(a[i].w);
                sum=sum-t+a[i].w;
            }
        }
    }
    printf("%d\n%lld",l,sum);
    return 0;
}
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值