农夫约翰有N元钱,市场上有价值1……K的商品无限个,求所有的花钱方案?
这是一个完全背包问题和大数的混合题目。
《2.3 记录结果再利用的“动态规划” 优化递推关系式》练习题的第三题。
定义
dp[i][j] := 用i种价格配出金额j的方案数。
那么dp[i][0] = 1,使用任何价格配出金额0的方案个数都是1(什么都不用)。
递推关系式:
dp[i][j] = dp[i – 1][j] + dp[i – 1][j – i] + dp[i – 1][j – 2 * i] + … + dp[i – 1][0]
解释一下,dp[i][j] += 所有dp[i – 1][x]之和,其中x为比j小i的整数倍的数。我这种“中文伪码”描述得很别扭,拿用例举个栗子吧:
dp[2][5] = dp[1][5] + dp[1][3] + dp[1][1];
于是初步的代码:
#include <iostream> using namespace std; unsigned long long dp[100 + 16][1000 + 16]; // dp[i][j] := 用i种价格配出金额j的方案数 ///////////////////////////SubMain////////////////////////////////// int main(int argc, char *argv[]) { #ifndef ONLINE_JUDGE freopen("in.txt", "r", stdin); freopen("out.txt", "w", stdout); #endif int N, K; cin >> N >> K; dp[0][0] = 1; for (int i = 1; i <= K; ++i) { for (int k = 0; k <= N; k += i) { for (int j = N; j >= k; --j) { dp[i][j] += dp[i - 1][j - k]; } } } cout << dp[K][N] << endl; #ifndef ONLINE_JUDGE fclose(stdin); fclose(stdout); system("out.txt"); #endif return 0; } ///////////////////////////End Sub//////////////////////////////////
上面这份代码提交后报WA,换用 unsigned long long 依然WA,猜测是大数。嘛,那就实现一个简易的大数加法吧,原本用一个 unsigned long long 的地方改用两个,分别表示低位和高位。由于 unsigned long long 最大范围到 1844674407370955161 ,所以取一个比这个值小一个数量级同时又能被10整除的数作为limit,得出下面这份AC代码:
#include <iostream> using namespace std; // 0高位 1低位 unsigned long long dp[100 + 16][1000 + 16][2]; // dp[i][j] := 用i种价格配出金额j的方案数 #define LIMIT_ULL 100000000000000000 ///////////////////////////SubMain////////////////////////////////// int main(int argc, char *argv[]) { #ifndef ONLINE_JUDGE freopen("in.txt", "r", stdin); freopen("out.txt", "w", stdout); #endif int N, K; cin >> N >> K; dp[0][0][1] = 1; for (int i = 1; i <= K; ++i) { for (int k = 0; k <= N; k += i) { for (int j = N; j >= k; --j) { dp[i][j][0] += dp[i - 1][j - k][0]; dp[i][j][1] += dp[i - 1][j - k][1]; // 高位进位 dp[i][j][0] += dp[i][j][1] / LIMIT_ULL; // 低位限制 dp[i][j][1] = dp[i][j][1] % LIMIT_ULL; } } } if (dp[K][N][0]) { cout << dp[K][N][0]; } cout << dp[K][N][1] << endl; #ifndef ONLINE_JUDGE fclose(stdin); fclose(stdout); system("out.txt"); #endif return 0; } ///////////////////////////End Sub//////////////////////////////////
12507590 | hankcs | 3181 | Accepted | 1856K | 500MS | C++ | 1025B | 2014-02-09 20:58:50 |
不过上面这份代码并非最优的,如果碰上了男人八题那种测试数据的话肯定会TLE吧。
为何说它不是最优的,因为在递推关系式中
dp[i][j] = dp[i – 1][j] + dp[i – 1][j – i] + dp[i – 1][j – 2 * i] + … + dp[i – 1][0]
将j换成j – i有
dp[i][j – i] = dp[i – 1][j – i] + dp[i – 1][j – 2 * i] + … + dp[i – 1][0]
于是就有:
dp[i][j] = dp[i – 1][j] + dp[i][j – i]
于是可以省掉一重循环,将复杂度降低到O(NK)。
……好冷,先去烤个火,指头冻僵了……
好的,优化递推关系式之后的代码:
#include <iostream> using namespace std; // 0高位 1低位 unsigned long long dp[100 + 16][1000 + 16][2]; // dp[i][j] := 用i种价格配出金额j的方案数 #define LIMIT_ULL 100000000000000000 ///////////////////////////SubMain////////////////////////////////// int main(int argc, char *argv[]) { #ifndef ONLINE_JUDGE freopen("in.txt", "r", stdin); freopen("out.txt", "w", stdout); #endif int N, K; cin >> N >> K; for (int i = 1; i <= K; ++i) { dp[i][0][1] = 1; for (int j = 1; j <= N; ++j) { if (j < i) { dp[i][j][0] = dp[i - 1][j][0]; dp[i][j][1] = dp[i - 1][j][1]; } else { dp[i][j][0] = dp[i - 1][j][0] + dp[i][j - i][0]; dp[i][j][1] = dp[i - 1][j][1] + dp[i][j - i][1]; // 高位进位 dp[i][j][0] += dp[i][j][1] / LIMIT_ULL; // 低位限制 dp[i][j][1] = dp[i][j][1] % LIMIT_ULL; } } } if (dp[K][N][0]) { cout << dp[K][N][0]; } cout << dp[K][N][1] << endl; #ifndef ONLINE_JUDGE fclose(stdin); fclose(stdout); system("out.txt"); #endif return 0; } ///////////////////////////End Sub//////////////////////////////////
这份代码速度比上一份快了一个数量级。
12507823 | hankcs | 3181 | Accepted | 1856K | 16MS | C++ | 1108B | 2014-02-09 22:24:47 |
知识共享署名-非商业性使用-相同方式共享:码农场 » POJ 3181 Dollar Dayz 题解 《挑战程序设计竞赛(第2版)》
看了你写的两篇dp的题解,从思路的分析到优化讲的都十分清楚!!感谢
dp += 所有dp 这里有个笔误 是 i-1.
感谢指正
谢谢!看书理解那个优化递推方程式很久没有彻悟,今天看了真是醍醐灌顶!