AcWing:145. 超市(贪心 + 小根堆 or 贪心 + 并查集)
超市里有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 贪心 + 并查集)的更多相关文章
- AcWing:111. 畜栏预定(贪心 + 小根堆)
有N头牛在畜栏中吃草. 每个畜栏在同一时间段只能提供给一头牛吃草,所以可能会需要多个畜栏. 给定N头牛和每头牛开始吃草的时间A以及结束吃草的时间B,每头牛在[A,B]这一时间段内都会一直吃草. 当两头 ...
- UOJ356 [JOI2017春季合宿] Port Facility 【启发式合并】【堆】【并查集】
题目分析: 好像跑得很快,似乎我是第一个启发式合并的. 把玩具看成区间.首先很显然如果有两个玩具的进出时间有$l1<l2<r1<r2$的关系,那么这两个玩具一定在不同的栈中间. 现在 ...
- AcWing:146. 序列(小根堆 + 数学归纳 + 贪心)
给定m个序列,每个包含n个非负整数. 现在我们可以从每个序列中选择一个数字以形成具有m个整数的序列. 很明显,我们一共可以得到nmnm个这种序列, 然后我们可以计算每个序列中的数字之和,并得到nmnm ...
- [CSP-S模拟测试]:梦境(贪心+小根堆)
题目描述 智者奥尔曼曾说过:有缘的人即使相隔海角天涯,也会在梦境中相遇. $IcePrince\text{_}1968$和$IcePrincess\text{_}1968$便是如此.有一天$IcePr ...
- 【BZOJ4388】JOI2012 invitation 堆+线段树+并查集(模拟Prim)
[BZOJ4388]JOI2012 invitation Description 澳洲猴举办了一场宴会,他想要邀请A个男生和B个女生参加,这A个男生从1到A编号,女生也从1到B编号.现在澳洲猴知道n组 ...
- AcWing:238. 银河英雄传说(带权并查集)
有一个划分为N列的星际战场,各列依次编号为1,2,…,N. 有N艘战舰,也依次编号为1,2,…,N,其中第i号战舰处于第i列. 有T条指令,每条指令格式为以下两种之一: 1.M i j,表示让第i号战 ...
- POJ 1456 Supermarket(贪心+并查集)
题目链接:http://poj.org/problem?id=1456 题目大意:有n件商品,每件商品都有它的价值和截止售卖日期(超过这个日期就不能再卖了).卖一件商品消耗一个单位时间,售卖顺序是可以 ...
- bzoj 1455 可并堆+并查集
一个堆和一个并查集对应,并且满足并查集中所有没有死的人等于堆中的人 /************************************************************** Pr ...
- bzoj 1577: [Usaco2009 Feb]庙会捷运Fair Shuttle——小根堆+大根堆+贪心
Description 公交车一共经过N(1<=N<=20000)个站点,从站点1一直驶到站点N.K(1<=K<=50000)群奶牛希望搭乘这辆公交车.第i群牛一共有Mi(1& ...
随机推荐
- Linux下安装Jenkins并且发布.net core
一,基础环境 1,操作系统:CentOS 7.3 2,Docker version 18.09.6 docker安装参考:https://www.cnblogs.com/liuxiaoji/p/110 ...
- vs 2013 设置website项目端口
vs 2015/2013 设置website项目端口 在web项目创建之后,当我想重新debug时,出现the port xxx is in use 错误. 经过netstat分析,发现占用此项目端口 ...
- JS基础_if练习三
<!DOCTYPE html> <html> <head> <meta charset="utf-8" /> <title&g ...
- springboot-oracle工程win下正常,centos下不能访问数据库
工程在win下正常运行,部署到centos下出现下述异常: ### Error querying database. Cause: org.springframework.jdbc.CannotGet ...
- 转:GitHub团队项目合作流程
转自:https://www.cnblogs.com/schaepher/p/4933873.html GitHub团队项目合作流程 已在另一篇博客中写出关于以下问题的解决,点此进入: 同步团队项 ...
- echart tooltip问题(鼠标放上去显示所有和显示当个)
先看效果 两种方式只要修改一下 echat option里面tooltip的属性即可 第一种: tooltip : { show: true, trigger: 'item' // trigger: ...
- Flask框架学习篇(一)
安装好Python,pip install flask安装好flask后,开始编写第一个flask程序 #包含动态路由的flask程序from flask import Flask app= Flas ...
- Mac EI 10.11.3 MySQL5.7.11 .dmg 安装(便捷设置,密码重置,卸载)
MySQL 5.7+ 安装成功以后会弹出一个临时密码 后面需要通过临时密码设置新的密码 重置root密码:安装成功后,使用临时密码登陆:敲入命令,mysqladmin -u root -p passw ...
- 解决myeclipse没有代码提示的问题
今天和室友安装了一样的myeclipse版本,结果室友的自动提示功能有,我的输入“.”后却不能提示,这对我们敲代码简直来说是一个折磨,不能自动提示,本来还以为是系统问题,一个是win7,一个是win1 ...
- 在Linux中,当需要从磁盘读取块时,进程状态会发生什么变化?被封锁了吗?如果是这样,如何选择另一个流程来执行?
当某个进程需要从磁盘中获取数据时,它实际上会停止在CPU上运行以让其他进程运行,因为该操作可能需要很长时间才能完成-至少需要5ms的磁盘寻道时间,而5ms就是1000万从程序的角度来看,CPU周期是永 ...