有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<bits/stdc++.h>
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_queue<node>q;
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<<temp.c<<endl;
q.pop();
}
}
printf("%lld\n",ans);
}
return 0;
}