家庭作业(homework)

题面

轩轩有太多的作业要做啊!!!!!!!!为了能高效完成作业,规定每项
作业花一个单位时间。

他的学习日从 0 时刻开始,有 100000 个单位时间(!)。在任一时刻,他
都可以选择编号 1NN 项作业中的任意一项作业来完成。

因为他在每个单位时间里只能做一个作业,而每项作业又有一个截止日期,
所以他很难有时间完成所有 N 个作业,虽然还是有可能。
对于第 i 个作业,有一个截止时间 Di,如果他可以完成这个作业,那么他
可以获得分数 Pi.

在给定的作业分数和截止时间下,FJ 能够获得的分数最大为多少呢?答案
可能会超过 32 位整型。

输入格式

第 1 行:一个整数 N.

2N+1 行:第 i+1 行有两个用空格分开的整数:DiPi.

输出格式

输出一行,里面有一个整数,表示最大获分值。

数据范围与约定

对于前 20% 的数据,1N100.

对于前 40% 的数据,1N1000.

对于前 60% 的数据,1N20000.

对于 100% 的数据,1N100000
1Di1000001Pi1000000000

思路

1.朴素贪心

先对作业按 Di 从小到大排序,再按 Pi 从大到小排序。

然后直接模拟时间流逝,能做就做,不能做跳过。

时间复杂度 O(nlogn)

可惜这不是最优解。

2.正解

我觉得是进阶贪心。

先把作业按 Di 从大到小排序。

然后直接扫一遍,如果遇到时间够就时间--,然后答案弹出优先队列累加。最后时间重置成当前的Di,并将 Pi 塞进优先队列。

代码

#include<bits/stdc++.h>
#define int long long
using namespace std;

int n;

struct homework{
	int d,p;
	homework(){
		d=0,p=0;
	}
	bool operator<(const homework ano) const {
		return d>ano.d;
	}
} school [100005];
priority_queue<int> pq;
int ttime=100000,ans;

signed main(){
	cin>>n;
	for(int i=0;i<n;i++){
		cin>>school[i].d>>school[i].p;
	}
	sort(school,school+n+1);
	for(int i=0;i<=n;i++){
		while(ttime>school[i].d && !pq.empty()){
			ttime--;
			ans+=pq.top();
			pq.pop();
		}
		ttime=school[i].d;
		pq.push(school[i].p);
	}
	cout<<ans<<endl;
	return 0;
}

懒得打 std::freopen 了。

posted @   蒟蒻xiezheyuan  阅读(480)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 阿里最新开源QwQ-32B,效果媲美deepseek-r1满血版,部署成本又又又降低了!
· 单线程的Redis速度为什么快?
· SQL Server 2025 AI相关能力初探
· AI编程工具终极对决:字节Trae VS Cursor,谁才是开发者新宠?
· 展开说说关于C#中ORM框架的用法!
点击右上角即可分享
微信分享提示