Q: 既是多重背包, 还是找零问题, 怎么处理?

A: 题意理解有误, 店主支付的硬币没有限制, 不占额度, 所以此题不比 1252 难多少

Description

Farmer John has gone to town to buy some farm supplies. Being a very efficient man, he always pays for his goods in such a way that the smallest number of coins changes hands, i.e., the number of coins he uses to pay plus the number of coins he receives in change is minimized. Help him to determine what this minimum number is.

FJ wants to buy T (1 ≤ T ≤ 10,000) cents of supplies. The currency system has N (1 ≤ N ≤ 100) different coins, with values V1V2, ..., VN (1 ≤ Vi ≤ 120). Farmer John is carrying C1 coins of value V1C2 coins of value V2, ...., and CN coins of value VN (0 ≤Ci ≤ 10,000). The shopkeeper has an unlimited supply of all the coins, and always makes change in the most efficient manner (although Farmer John must be sure to pay in a way that makes it possible to make the correct change).

Input

Line 1: Two space-separated integers: N and T
Line 2: N space-separated integers, respectively V1V2, ..., VN coins (V1, ...VN
Line 3: N space-separated integers, respectively C1C2, ..., CN

Output

Line 1: A line containing a single integer, the minimum number of coins involved in a payment and change-making. If it is impossible for Farmer John to pay and receive exact change, output -1.

Sample Input

3 70
5 25 50
5 2 1

Sample Output

3

Hint

Farmer John pays 75 cents using a 50 cents and a 25 cents coin, and receives a 5 cents coin in change, for a total of 3 coins used in the transaction.

题意:

John 去付款, 手上拿着各种面值的硬币, 每种面值的硬币有一个可以使用的上限, 店主支持找钱, 找钱所用的硬币是无限的

思路:

1. 证明背包的容量上限是个难点, 这里有个证明

http://www.cppblog.com/Davidlrzh/articles/135614.html

背包容量上限是 T + vmax^2

2. 使用倍增优化转化为 0/1 背包来做. 先求解 Jhon 的给钱方式, 再求解找钱方案, 两次 DP

3.  Jhon 的状态转移方程. dp[i][j] = min(dp[i][j], dp[i][j-w[i]]+1)

   店主的状态转移方程可以有两种表述, dp2[i][j] = min(dp2[i][j], dp2[i][j-w[i]]+1) (和 Jhon 的一样), 但是用到两个dp 数组

4. 第二种表述和 1252 相同, 仅使用一个dp 数组, 总的状态转移方程为 dp[j] = min([j-w[i]+1, dp[j+w[i]]+1);

总结:

1. 看别人的解题报告, 才知道店主找零的硬币是没有上限的, 这将题目的难度下降了很多, 难点就变成多重背包的优化了

2. 思路(3,4)的状态转移方程有误, 因为进行过倍增优化, 所以状态转移方程中并不是简单的+1, 而应该+倍增优化的倍数

3. 没判断输出为 INF 的情况, WA 了一次, 找到了测试用例地址

http://cerberus.delos.com:790/TESTDATA/DEC06_5.htm

代码:

#include <iostream>
using namespace std; int N,T;
int V[150], C[150];
int upload;
int dp[100000];
int stuff[10010];
int cnt[10010];
int len;
const int INF = 0X3F3F3F3F; /*
* 倍增优化转成 01 背包
* 处理 01 背包, 求解 Jhon 恰好付 v 元 所需要的硬币数
*/
void firstPass() {
len = 0;
for(int i = 0; i < N; i ++) {
int rem = C[i];
int j = 0; while(rem) { // 倍增优化转化成 01 背包
if(rem >= (1<<j)) {
stuff[len++] = V[i]*(1<<j);
cnt[len-1] = (1<<j);
rem -= (1<<j);
j++;
}else{
stuff[len++] = V[i]*rem;
cnt[len-1] = rem;
rem = 0;
}
}
} memset(dp, 0x3f, sizeof(int)*(upload+10));
dp[0] = 0;
for(int i = 0; i < len; i ++) {
for(int v = upload; v >= stuff[i]; v--) {
dp[v] = min(dp[v], dp[v-stuff[i]]+cnt[i]);
//printf("dp[%d], stuff[%d] = %d\n", v, i, stuff[i]);
}
}
} /*
* second pass 完全背包
* 状态转移方程, dp[v] = min(dp[v], dp[v+w[i]]+1)
* 要注意遍历顺序
*/
int secondPass() {
for(int i = 0; i < N; i ++) {
for(int v = upload-V[i]; v >= 0; v --) {
dp[v] = min(dp[v], dp[v+V[i]]+1);
}
}
return dp[T];
}
int main() {
freopen("E:\\Copy\\ACM\\测试用例\\in.txt", "r", stdin);
cin >> N >> T;
int vmax = 0;
int sum = 0;
for(int i = 0; i < N; i ++) {
scanf("%d", &V[i]);
vmax = max(vmax, V[i]);
}
upload = vmax*vmax + T; for(int i = 0; i < N; i ++) {
scanf("%d", &C[i]);
sum += C[i]*V[i];
}
if(sum < T)
cout << -1 << endl;
else {
firstPass();
int ans = secondPass();
if(ans != INF)
cout << secondPass() << endl;
else
cout << -1 << endl;
} return 0;
}

  

POJ 3260 The Fewest Coins(多重背包问题, 找零问题, 二次DP)的更多相关文章

  1. POJ 3260 The Fewest Coins(多重背包+全然背包)

    POJ 3260 The Fewest Coins(多重背包+全然背包) http://poj.org/problem?id=3260 题意: John要去买价值为m的商品. 如今的货币系统有n种货币 ...

  2. POJ 3260 The Fewest Coins(背包问题)

    [题目链接] http://poj.org/problem?id=3260 [题目大意] 给出你拥有的货币种类和每种的数量,商店拥有的货币数量是无限的, 问你买一个价值为m的物品,最少的货币流通数量为 ...

  3. POJ 3260 The Fewest Coins(完全背包+多重背包=混合背包)

    题目代号:POJ 3260 题目链接:http://poj.org/problem?id=3260 The Fewest Coins Time Limit: 2000MS Memory Limit: ...

  4. poj 3260 The Fewest Coins

    // 转载自http://blog.163.com/benz_/blog/static/18684203020115721917109/算法不难看出,就是一个无限背包+多重背包.问题在于背包的范围.设 ...

  5. POJ 1252 Euro Efficiency(完全背包, 找零问题, 二次DP)

    Description On January 1st 2002, The Netherlands, and several other European countries abandoned the ...

  6. POJ 3260 The Fewest Coins 最少硬币个数(完全背包+多重背包,混合型)

    题意:FJ身上有各种硬币,但是要买m元的东西,想用最少的硬币个数去买,且找回的硬币数量也是最少(老板会按照最少的量自动找钱),即掏出的硬币和收到的硬币个数最少. 思路:老板会自动找钱,且按最少的找,硬 ...

  7. POJ3260——The Fewest Coins(多重背包+完全背包)

    The Fewest Coins DescriptionFarmer John has gone to town to buy some farm supplies. Being a very eff ...

  8. codevs 3961 硬币找零【完全背包DP/记忆化搜索】

    题目描述 Description 在现实生活中,我们经常遇到硬币找零的问题,例如,在发工资时,财务人员就需要计算最少的找零硬币数,以便他们能从银行拿回最少的硬币数,并保证能用这些硬币发工资. 我们应该 ...

  9. [LeetCode] Coin Change 2 硬币找零之二

    You are given coins of different denominations and a total amount of money. Write a function to comp ...

随机推荐

  1. tar压缩解压缩命令详解

    tar命令详解 -c: 建立压缩档案 -x:解压 -t:查看内容 -r:向压缩归档文件末尾追加文件 -u:更新原压缩包中的文件 这五个是独立的命令,压缩解压都要用到其中一个,可以和别的命令连用但只能用 ...

  2. win32串口编程

    翻译自:ms-help://MS.MSDNQTR.v80.chs/MS.MSDN.v80/MS.WIN32COM.v10.en/dnfiles/html/msdn_serial.htm 老外写的文章, ...

  3. vue使用sweetalert2弹窗插件

    1). 安装 sweetalert2 npm install sweetalert2@7.15.1 --save 2). 封装 sweetalert2 在 src 新建 plugins 文件夹,然后新 ...

  4. dubbo注册zookeeper保错原因

    我的zookeeper是安装在本地,用的默认端口2181,版本3.4.10.dubbo版本2.5.8.dubbo-demo-provider.xml配置文件修改为:<dubbo:registry ...

  5. C++面向对象程序设计的一些知识点(3)

    摘要:多态性提供一组统一的调用接口函数,依据这些条用接口函数具体对象的不同,同一名字的函数会有不同的行为. 1.重载与隐藏 (1).对同一作用域中的同名函数,如果它们的函数特征标不同,那么它们就形成一 ...

  6. php写杨辉三角算法

    <?phpfunction YangHui($iLine) {      for ($i = 0;$i <= $iLine;$i++)//行      {         for ($j ...

  7. 寻找最大的k个数问题

    这是编程之美书第2.5节的一道题目. 各种解法: 解法一,用nlgn复杂度的排序算法对数组进行从大到小排序,取前K个.但这方法做了两件不必要做的事:它对想得到的K个数进行了排序,对不想得到的n-K个数 ...

  8. C++笔记 3

    1.数组是自动分配空间,指针要手工分配空间(int *p = new int;) 2.在Unix上,程序出现段错误的时候,系统会生成core 文件,会把出现错误的那一刻的程序镜像保存在此文件中 3.结 ...

  9. C++基础学习-20120516

    1.一下是使用strcpy_s与strcpy的安全性比较 char szBuf[2] = {0}; strcpy_s(szBuf, 2, "12131");  //新的CRT函数  ...

  10. ReentrantReadWriteLock锁例子

    锁所提供的最重要的改进之一就是ReadWriteLock接口和唯一 一个实现它的ReentrantReadWriteLock类.这个类提供两把锁,一把用于读操作和一把用于写操作.同时可以有多个线程执行 ...