超市里有N件商品,每个商品都有利润pipi和过期时间didi,每天只能卖一件商品,过期商品(即当天di<=0di<=0)不能再卖。

求合理安排每天卖的商品的情况下,可以得到的最大收益是多少。

输入格式

输入包含多组测试用例。

每组测试用例,以输入整数N开始,接下里输入N对pipi和didi,分别代表第i件商品的利润和过期时间。

在输入中,数据之间可以自由穿插任意个空格或空行,输入至文件结尾时终止输入,保证数据正确。

输出格式

对于每组产品,输出一个该组的最大收益值。

每个结果占一行。

数据范围

0≤N≤100000≤N≤10000,
1≤pi,di≤100001≤pi,di≤10000

输入样例:

4  50 2  10 1   20 2   30 1

7  20 1   2 1   10 3  100 2   8 2
5 20 50 10

输出样例:

80
185

算法:贪心 + 小根堆

题解:根据题意,一天只能卖一个商品,那么,我们就可以用贪心的思想来实现,用小根堆(可以用优先队列来实现,不一定要手写)来维护最大值。

优先队列实现小根堆:

#include <iostream>
#include <cstdio>
#include <queue>
#include <functional>
#include <algorithm> using namespace std; const int maxn = 1e5+; struct node {
int pi, di;
}arr[maxn]; priority_queue<int, vector<int>, greater<int> > p; //建立从小到大的优先队列(小根堆) bool cmp(node a, node b) {
return a.di < b.di;
} int main() {
int n;
while(~scanf("%d", &n)) {
for(int i = ; i <= n; i++) {
scanf("%d %d", &arr[i].pi, &arr[i].di);
}
sort(arr + , arr + n + , cmp);
for(int i = ; i <= n; i++) {
if(arr[i].di > p.size()) { //如果商品的过期天数大于当前天数,就直接进入队列
p.push(arr[i].pi);
} else if(arr[i].di == p.size() && arr[i].pi > p.top()) { //当商品的过去天数等于当前天数的时候,就比较一下当前商品是否比队列中最小的商品大,如果是就进入队列
p.pop();
p.push(arr[i].pi);
}
}
int ans = ;
while(!p.empty()) {
ans += p.top();
p.pop();
}
printf("%d\n", ans);
}
return ;
}

二叉堆实现小根堆:

#include <iostream>
#include <cstdio>
#include <algorithm> using namespace std; const int maxn = 1e5+; struct node {
int pi, di;
}arr[maxn];
int heap[maxn];
int tot; bool cmp(node a, node b) {
return a.di < b.di;
} void up(int n) {
while(n > ) {
if(heap[n] < heap[n / ]) {
swap(heap[n], heap[n / ]);
}
n /= ;
}
} void insert(int val) {
heap[++tot] = val;
up(tot);
} void down(int n) {
int s = n * ;
while(s <= tot) {
if(s < tot && heap[s] > heap[s + ]) { //找出左右子树中较小的那个
s++;
}
if(heap[s] < heap[n]) {
swap(heap[s], heap[n]);
}
n = s;
s = * n;
} } void update(int val) {
heap[] = val;
down();
} int main() {
int n;
while(~scanf("%d", &n)) {
for(int i = ; i <= n; i++) {
scanf("%d %d", &arr[i].pi, &arr[i].di);
}
sort(arr + , arr + n + , cmp);
tot = ;
for(int i = ; i <= n; i++) {
if(arr[i].di > tot) {
insert(arr[i].pi);
} else if(arr[i].di == tot && arr[i].pi >heap[]) {
update(arr[i].pi);
}
}
int ans = ;
for(int i = ; i <= tot; i++) {
ans += heap[i];
}
printf("%d\n", ans);
}
return ;
}

并查集:

#include <iostream>
#include <cstdio>
#include <vector>
#include <algorithm> using namespace std; const int maxn = 1e4+; struct node {
int weight, date;
friend bool operator < (node a, node b) {
if(a.weight == b.weight) {
return a.date > b.date;
}
return a.weight > b.weight;
}
}; vector<node> v;
int f[maxn]; int find(int x) {
if(f[x] != x) {
return f[x] = find(f[x]);
}
return f[x];
} int main() {
int n;
while(~scanf("%d", &n)) {
v.clear();
int maxe = ;
for(int i = ; i < n; i++) {
int x, y;
scanf("%d %d", &x, &y);
v.push_back((node){x, y});
maxe = max(maxe, y);
}
for(int i = ; i <= maxe; i++) {
f[i] = i;
}
sort(v.begin(), v.end()); //按权值从大到小排序
int size = v.size();
int ans = ;
for(int i = ; i < size; i++) {
int w = v[i].weight;
int d = v[i].date;
int pos = find(d); //找到当前日期的最佳位置
if(pos > ) { //如果该位置没有出界,那么就可以使用
ans += w;
f[pos] = pos - ;
}
}
printf("%d\n", ans);
}
return ;
}

AcWing:145. 超市(贪心 + 小根堆 or 贪心 + 并查集)的更多相关文章

  1. AcWing:111. 畜栏预定(贪心 + 小根堆)

    有N头牛在畜栏中吃草. 每个畜栏在同一时间段只能提供给一头牛吃草,所以可能会需要多个畜栏. 给定N头牛和每头牛开始吃草的时间A以及结束吃草的时间B,每头牛在[A,B]这一时间段内都会一直吃草. 当两头 ...

  2. UOJ356 [JOI2017春季合宿] Port Facility 【启发式合并】【堆】【并查集】

    题目分析: 好像跑得很快,似乎我是第一个启发式合并的. 把玩具看成区间.首先很显然如果有两个玩具的进出时间有$l1<l2<r1<r2$的关系,那么这两个玩具一定在不同的栈中间. 现在 ...

  3. AcWing:146. 序列(小根堆 + 数学归纳 + 贪心)

    给定m个序列,每个包含n个非负整数. 现在我们可以从每个序列中选择一个数字以形成具有m个整数的序列. 很明显,我们一共可以得到nmnm个这种序列, 然后我们可以计算每个序列中的数字之和,并得到nmnm ...

  4. [CSP-S模拟测试]:梦境(贪心+小根堆)

    题目描述 智者奥尔曼曾说过:有缘的人即使相隔海角天涯,也会在梦境中相遇. $IcePrince\text{_}1968$和$IcePrincess\text{_}1968$便是如此.有一天$IcePr ...

  5. 【BZOJ4388】JOI2012 invitation 堆+线段树+并查集(模拟Prim)

    [BZOJ4388]JOI2012 invitation Description 澳洲猴举办了一场宴会,他想要邀请A个男生和B个女生参加,这A个男生从1到A编号,女生也从1到B编号.现在澳洲猴知道n组 ...

  6. AcWing:238. 银河英雄传说(带权并查集)

    有一个划分为N列的星际战场,各列依次编号为1,2,…,N. 有N艘战舰,也依次编号为1,2,…,N,其中第i号战舰处于第i列. 有T条指令,每条指令格式为以下两种之一: 1.M i j,表示让第i号战 ...

  7. POJ 1456 Supermarket(贪心+并查集)

    题目链接:http://poj.org/problem?id=1456 题目大意:有n件商品,每件商品都有它的价值和截止售卖日期(超过这个日期就不能再卖了).卖一件商品消耗一个单位时间,售卖顺序是可以 ...

  8. bzoj 1455 可并堆+并查集

    一个堆和一个并查集对应,并且满足并查集中所有没有死的人等于堆中的人 /************************************************************** Pr ...

  9. bzoj 1577: [Usaco2009 Feb]庙会捷运Fair Shuttle——小根堆+大根堆+贪心

    Description 公交车一共经过N(1<=N<=20000)个站点,从站点1一直驶到站点N.K(1<=K<=50000)群奶牛希望搭乘这辆公交车.第i群牛一共有Mi(1& ...

随机推荐

  1. join函数详解

    定义:join() 方法用于把数组中的所有元素放入一个字符串. 语法 : ArrayObject.join(separator)   separator 可选.指定要使用的分隔符.如果省略该参数,则使 ...

  2. Presto基础知识

    背景 MapReduce不能满足大数据快速实时adhoc查询计算的性能要求. Facebook的数据仓库存储在少量大型Hadoop/HDFS集群.Hive是Facebook在几年前专为Hadoop打造 ...

  3. sqlserver关于发布订阅replication_subscription的总结

    (转载)sqlserver关于发布订阅replication_subscription的总结 来自 “ ITPUB博客 ” ,原文地址:http://blog.itpub.net/30126024/v ...

  4. 分布式缓存Redis+Memcached经典面试题和答案

    Redis相比memcached有哪些优势? (1) memcached所有的值均是简单的字符串,redis作为其替代者,支持更为丰富的数据类型 (2) redis的速度比memcached快很多 ( ...

  5. Redis-String常用命令

    Redis-String常用命令 set key value- 设置Key-value键值对 get key 获取指定key对应的值 append key value 在指定key对应值的后面追加va ...

  6. centos 7 安装 LNMPC cacti 1.2.7 监控

    先上图,后续更新

  7. python常用模块:logging、hashlib、re

    今日内容主要有:一.logging模块二.logging模块的使用三.hashlib模块四.re模块 一.logging模块 import logging # 1 日志的级别 logging.debu ...

  8. Java程序向MySql数据库中插入的中文数据变成了问号

    找到mysql的安装目录,修改my.ini文件 (1)如何找到my.ini文件 如果my.ini文件不在MySQL的安装目录下,可能放在隐藏目录,要先去找到ProgramData,(这里要先打开显示隐 ...

  9. 【XDOJ】小W的塔防

    原题: 小W在成功拿到iPhone后,下载了一个塔防游戏.游戏的目标是阻止僵尸穿过地图. 地图可以看作一条长度为n的线段,这条线段被划分为n条单位长度的小线段.僵尸需要花费t秒才能通过一条小线段.在每 ...

  10. mongodb单机搭建

    参考网站:http://www.runoob.com/mongodb/mongodb-linux-install.html 1.下载 https://www.mongodb.com/download- ...