51 nod 最高的奖励

有N个任务,每个任务有一个最晚结束时间以及一个对应的奖励。在结束时间之前完成该任务,就可以获得对应的奖励。完成每一个任务所需的时间都是1个单位时间。有时候完成所有任务是不可能的,因为时间上可能会有冲突,这需要你来取舍。求能够获得的最高奖励。
Input
第1行:一个数N,表示任务的数量(2 <= N <= 50000)
第2 - N + 1行,每行2个数,中间用空格分隔,表示任务的最晚结束时间E[i]以及对应的奖励W[i]。(1 <= E[i] <= 10^9,1 <= W[i] <= 10^9)
Output
输出能够获得的最高奖励。
Input示例
7
4 20
2 60
4 70
3 40
1 30
4 50
6 10
Output示例
230

http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1163

分析一下这个题哈,首先每一个单位时间只能做一件事情,对于样例来说,我们其实就是考虑结束时间为4的三种情况我们怎么取而已吧。

你会发现,我们考虑4的情况,我们只会去和结束时间为1,2,3,的来讨论,对于结束时间是6的,我们是不会管的,所以6我们是一定会取的。因为在那个时间

没有事情会和它冲突。


所有我们就应该倒着来讨论。

因为这个最大的时间是10^9次方,太大了,我们可以把这个时间缩小,缩小到n就可以了,大于n的就让他直接等于n,这样考虑是不会影响最终的结果的。

因为最多就是n件事情,每个一个单位时间。



然后我们就倒着找,当时间为n的时候,把所有时间为n的找出来,取一个最大值加起来,怎么快速取到最大?

优先队列。就可以咯。

#include
using namespace std;
struct node
{
    int e;
    long long c;
        bool operator<(const node &C)const{
            return C.c>c;
        }
} a[50005];
bool cmp(node aa,node bb)
{
    if(aa.e==bb.e)
        return aa.c>bb.c;
    return aa.e>bb.e;
}
int main()
{
    int n;
    while(~scanf("%d",&n))
    {
        priority_queueq;
        for(int i=1; i<=n; i++)
        {
            int x;
            scanf("%d%lld",&x,&a[i].c);
            a[i].e=min(x,n);
        }
        sort(a+1,a+n+1,cmp);
//        for(int i=1;i<=n;i++)
//            printf("%d %d\n",a[i].e,a[i].c);
        int j=1;
        long long ans=0;
        for(int i=n;i>=1;i--)
        {
            while(a[j].e>=i&&j<=n)
            {
                //printf("j=%d\n",j);
                q.push(a[j]);
                j++;
            }
            if(!q.empty())
            {
                node temp=q.top();
                ans+=temp.c;
                cout<



 
  


你可能感兴趣的:(模拟)