题目描述
伴随着科技的发展,我们的生活也越来越多姿多彩,随着手机的普及,各种交友软件也在快速的发展。
小a是个老实人,当然只是自己理解而已,其实小a是个不折不扣的渣男。因为他在有女朋友的同时,还在疯狂的撒网,利用各种交友软件寻求更适合自己的伴侣。
小a女朋友当然不是省油的灯,自然了解小a的本性,所以在每次见面时就会翻看小a的软件记录,来了解小a近期的状况,机智的小a当然会在女朋友来之前就给完全清理干净了。
终于在某次时间紧急的情况下,小a的软件记录并不一定能够完全删除,但是小a知道,自己每个软件记录的火热程度以及最终删除时间(即可以删除的最晚时间,过时将无法删除)。每个软件记录的删除需要一分钟,软件记录的火热程度,正好对应着女朋友的刺激值,小a想知道,在有限的时间内,如何才能够让女朋友的刺激值最小,求出最小值。
输入
第一行一个正整数T。表示样例个数(0 < T < 10)
每组有两个整数n,m,分别表示一共有n个软件以及女朋友到来的时间(0 < n <= 10^5,0 < m <= 10^6)
往下对应着n行,每行有两个正整数e,f分别对应每个软件记录的火热程度和该软件的最终删除时间。(0 < e <= 10^5,0 < f <= 10^6)
题目中涉及到的时间全部以分钟为单位。
输出
输出对女朋友的最小刺激值;结果占一行。
样例输入
2
4 2
20 1
10 1
30 2
40 2
6 2
20 1
10 1
30 2
40 2
50 3
60 3
样例输出
30
100
思路
- 贪心: 时间从后往前找最大的刺激的值。
- 优先队列,数组模拟超时
AC
#include<bits/stdc++.h>
#define ll long long
#define mem(a, b) memset(a, b, sizeof(a))
#define N 100005
#define P pair<ll, int>
using namespace std;
struct ac{
ll v, t;
//优先队列中 刺激值降序
bool operator < (const ac &x) const{
return x.v > v;
}
}a[N];
// 时间降序
bool cmp_t (ac x, ac y) {
return x.t > y.t;
}
int main() {
// freopen("in.txt", "r", stdin);
int t;
scanf("%d", &t);
while (t--) {
priority_queue<ac, vector<ac> >que;
int n, m;
//sum 总刺激值
//ans 先记录可以消除的刺激值;sum - ans
int sum = 0, ans = 0;
scanf("%d%d", &n, &m);
for (int i = 0; i < n; i++) {
scanf("%d%d", &a[i].v, &a[i].t);
sum += a[i].v;
}
sort(a, a + n, cmp_t);
int now = 0;
//时间从后往前
while (m) {
while (now < n && a[now].t >= m) {
que.push(a[now]);
now++;
}
//每次去队列中的最大值
if (!que.empty()) {
ans += que.top().v;
que.pop();
}
m--;
}
ans = sum - ans;
printf("%d\n", ans);
}
return 0;
}