挑战程序设计竞赛 2.2 poj 3040 Allowance 贪心
https://vjudge.csgrandeur.cn/problem/POJ-3040
/*
作为创纪录的牛奶产量的奖励,约翰决定每周给贝西一小笔零用钱。FJ拥有一组N(1 <= N <= 20)种不同面额的硬币,
其中每个面额的硬币均可整除较大面额的硬币(例如,1分硬币、5分硬币、10分硬币和50分硬币)。
使用给定的硬币面额,他想每周至少支付给贝西一定金额C(1 <= C <= 100,000,000)。
请帮助他计算他最多可以支付贝西多少周。
输入
第1行:两个以空格分隔的整数:N和C
第2行到第N+1行:每行对应一个硬币面额,并包含两个整数:
硬币面额V(1 <= V <= 100,000,000)和约翰手中该面额的硬币数量B(1 <= B <= 1,000,000)。
输出
第1行:一个整数,表示约翰可以支付贝西至少C零用钱的周数。
3 6
10 1
1 100
5 120
111
3 6
10 3
20 4
40 5
12
3 51
100 1
50 4
1 2
4
3 51
1 2
50 4
100 1
4
20 100000000
67108864 1000000
33554432 1000000
16777216 1000000
8388608 1000000
4194304 1000000
2097152 1000000
1048576 1000000
524288 1000000
262144 1000000
131072 1000000
65536 1000000
32768 1000000
16384 1000000
8192 1000000
4096 1000000
2048 1000000
1024 1000000
512 1000000
256 1000000
128 1000000
1340054
输入详情:
FJ每周想支付给贝西6美分。他手上有100个1美分硬币,120个5美分硬币和1个10美分硬币。
输出详情:
FJ可以用一个10美分硬币多支付给贝西1周,然后用两个5美分硬币支付给贝西10周,
最后用一个1美分硬币和一个5美分硬币支付给贝西100周。
*/
解答
直觉分析如下:
因为可选择的美分硬币数值是可整除的。所以我们需要尽量选择面额更大的硬币.
1 因为面值小的硬币总能替代面额大的硬币,更优。所以我们选择次优的较大面额的硬币,将更优的选择留给后面。
2 同样的 凑齐刚好等于每周报酬的面值能留下更多的硬币数值给后面的选择,所以优先选择刚好等于每周报酬的组合,
然后再选择最接近、大于等于每周报酬的组合。
代码如下
#include <iostream>
#include <cstring>
#include <map>
using namespace std;
int n, c;
int ans;
map<int, int> mm;
map<int, int> usedmm;
int limit;
void GetusedArr() {
//计算每次选择硬币的组合 逆序从大到小选择, 优先选择大额的 最接近等于每周报酬的组合
for (map<int, int>::reverse_iterator it = mm.rbegin(); it != mm.rend(); it++) {
if (it->second != 0 && limit >0 && limit >= it->first) {
int used = min(limit / it->first, it->second);
limit -= used * it->first;
usedmm[it->first] += used;
}
//剩余值为0 则说明抽出了刚好等于每周报酬的组合
if (limit == 0) break;
}
//贪心完所有金币 还没超出需要的金额. 说明凑不出刚好等于报酬的组合
// 从小到大选择硬币,使用较小的面值进行填充 最接近等于每周报酬
if (limit > 0) {
for (map<int, int>::iterator it = mm.begin(); it != mm.end(); it++) {
if (it->second > 0 && limit > 0 && mm[it->first] > usedmm[it->first]) {
int used = 1; int v = it->first; int cnt = it->second;
limit -= v;
usedmm[v] += 1;
}
}
}
return;
}
void solve() {
while (1) {
usedmm.clear();
limit = c;
GetusedArr();
//limit 还不等于小于0 那么说明现在的硬币已经不能凑齐一周报酬了
if (limit > 0) {
cout << ans << endl; return;
}
//按照usedmm 查看能取几次
int minget = 1000010;
for (map<int, int>::iterator it = usedmm.begin(); it != usedmm.end(); it++) {
int v = it->first;
minget = min(minget, mm[v]/it->second);
}
ans += minget;
//从已有的硬币里减去
for (map<int, int>::iterator it = usedmm.begin(); it != usedmm.end(); it++) {
int v = it->first;
mm[v] -= minget * it->second;
}
}
}
int main()
{
cin >> n >> c;
for (int i = 0; i < n; i++) {
int v, b; cin >> v >> b;
if (v >= c) {
ans += b;
}
else {
mm[v] += b;
}
}
solve();
return 0;
}
挑战程序设计竞赛 2.2 poj 3040 Allowance 贪心的更多相关文章
- POJ 3040 Allowance 贪心
这题目的贪心思路还是有一点细节问题的. 还没有证明,据说是因为题目给的条件是每个价格是比它小的价格的倍数才能这么贪心的. 思路如下: 假设要给奶牛的钱为C 1)从大面值到小面值一次拿钱,能拿多少拿多少 ...
- POJ 2386 Lake Counting 题解《挑战程序设计竞赛》
地址 http://poj.org/problem?id=2386 <挑战程序设计竞赛>习题 题目描述Description Due to recent rains, water has ...
- 《挑战程序设计竞赛》2.3 动态规划-优化递推 POJ1742 3046 3181
POJ1742 http://poj.org/problem?id=1742 题意 有n种面额的硬币,面额个数分别为Ai.Ci,求最多能搭配出几种不超过m的金额? 思路 据说这是传说中的男人8题呢,对 ...
- Aizu 2249Road Construction 单源最短路变形《挑战程序设计竞赛》模板题
King Mercer is the king of ACM kingdom. There are one capital and some cities in his kingdom. Amazin ...
- 挑战程序设计竞赛》P345 观看计划
<挑战程序设计竞赛>P345 观看计划 题意:一周一共有M个单位的时间.一共有N部动画在每周si时 ...
- poj 3253 Fence Repair 贪心 最小堆 题解《挑战程序设计竞赛》
地址 http://poj.org/problem?id=3253 题解 本题是<挑战程序设计>一书的例题 根据树中描述 所有切割的代价 可以形成一颗二叉树 而最后的代价总和是与子节点和深 ...
- POJ 3040 Allowance【贪心】
POJ 3040 题意: 给奶牛发工资,每周至少 C 元.约翰手头上有面值V_i的硬币B_i个,这些硬币的最小公约数为硬币的最小面值.求最多能发几周? 分析: 贪心策略是使多发的面额最小(最优解).分 ...
- 《挑战程序设计竞赛》2.2 贪心法-其它 POJ3617 3069 3253 2393 1017 3040 1862 3262
POJ3617 Best Cow Line 题意 给定长度为N的字符串S,要构造一个长度为N的字符串T.起初,T是一个空串,随后反复进行下列任意操作: 从S的头部(或尾部)删除一个字符,加到T的尾部 ...
- 【网络流#8】POJ 3469 Dual Core CPU 最小割【ISAP模板】 - 《挑战程序设计竞赛》例题
[题意]有n个程序,分别在两个内核中运行,程序i在内核A上运行代价为ai,在内核B上运行的代价为bi,现在有程序间数据交换,如果两个程序在同一核上运行,则不产生额外代价,在不同核上运行则产生Cij的额 ...
- <挑战程序设计竞赛> poj 3320 Jessica's Reading Problem 双指针
地址 http://poj.org/problem?id=3320 解答 使用双指针 在指针范围内是否达到要求 若不足要求则从右进行拓展 若满足要求则从左缩减区域 代码如下 正确性调整了几次 然后 ...
随机推荐
- celery笔记九之task运行结果查看
本文首发于公众号:Hunter后端 原文链接:celery笔记九之task运行结果查看 这一篇笔记介绍一下 celery 的 task 运行之后结果的查看. 前面我们使用的配置是这样的: # sett ...
- Android实时获取摄像头画面传输至PC端
前言 最近在做一个PC端小应用,需要获取摄像头画面,但是电脑摄像头像素太低,而且位置调整不方便,又不想为此单独买个摄像头.于是想起了之前淘汰掉的手机,成像质量还是杠杠的,能不能把手机摄像头连接到电脑上 ...
- Linux相关概念及操作
目录 linux的文件系统是采用级层式的树状目录结构,在此结构中的最上层是根目录"/",然后在此目录下再创建其他的目录. 1./bin 是Binary的缩写,这个目录存放着最经常使 ...
- 【原创】Ftrace使用及实现机制
Ftrace使用及实现机制 版权声明:本文为本文为博主原创文章,转载请注明出处 https://www.cnblogs.com/wsg1100 如有错误,欢迎指正. 目录 Ftrace使用及实现机制 ...
- [Spring+SpringMVC+Mybatis]框架学习笔记(九):Mybatis主配置文件和映射文件
第9章 Mybatis主配置文件和映射文件 9.1 用Mybatis进行开发的两种方式 在正式的开发环境中用Mybatis进行开发有两种方式: 1)原始的接口和实现类的方式 缺点: 重复代码太多,sq ...
- nmcli 命令简单使用
centos7/8 机器上默认有安装nmcli,可直接使用修改ip. nmcli c add type ethernet ifname eth0 # 会提示 'ethernet-eth0' 创建成功 ...
- 三个编程思想:面向对象编程、面向接口编程、面向过程编程【概念解析系列_1】【C# 基础】
〇.前言 对于 .Net 中的编程思想还是十分重要的,也是编码出高效的程序的基础! 在使用之前了解其本质,那么用起来就游刃有余.下面来简单对比下三个编程思想,看下它们都是什么,它们之间又有什么关系. ...
- python教程 入门学习笔记 第1天 初识python python语言环境安装 python编写器
初识python 一.python语言简介: 1.起源:1989年由荷兰的前谷歌程序员吉多.范罗苏姆(龟叔)创造,python的命名来源于英国电视喜剧Monty Python's Flying Cir ...
- html标签tr td是什么意思
<table>代表表格</table><tr>代表表格中的一行</tr><td>代表表格中的一列</td>'tr'与'td'交成 ...
- gitlab-runner-config-in-docker
gitlab in docker 网上有很多现成的解决方案,本文仅作流程梳理,若不需要,可直接用gitlab官方提供的镜像 installation Dockerfile FROM registry. ...