鲜花( 0) 鸡蛋( 0)
|
动态规划的基本思想是将待求解问题分解成若干个子问题,先求解子问题,并将这些子问题的解保存起来,如果以后在求解较大子问题的时候需要用到这些子问题的解,就可以直接取出这些已经计算过的解而免去重复运算。保存子问题的解可以使用填表方式,例如保存在数组中。
( O4 j7 D0 B6 u+ t8 q! T 用一个实际例子来体现动态规划的算法思想——硬币找零问题。
/ P: b3 ?( a2 f r 硬币找零问题描述:现存在一堆面值为 V1、V2、V3 … 个单位的硬币,问最少需要多少个硬币才能找出总值为 T 个单位的零钱?假设这一堆面值分别为 1、2、5、21、25 元,需要找出总值 T 为 63 元的零钱。
# D+ Q; A" Q. F- ~$ j3 i6 Y5 s 很明显,只要拿出 3 个 21 元的硬币就凑够了 63 元了。( A4 b9 q$ Q$ s- c* B
基于上述动态规划的思想,我们可以从 1 元开始计算出最少需要几个硬币,然后再求 2 元、3元…每一次求得的结果都保存在一个数组中,以后需要用到时则直接取出即可。那么我们什么时候需要这些子问题的解呢?如何体现出由子问题的解得到较大问题的解呢?1 |% n7 V8 f' \% L% t$ E: g6 q
其实,在我们从 1 元开始依次找零时,可以尝试一下当前要找零的面值(这里指 1 元)是否能够被分解成另一个已求解的面值的找零需要的硬币个数再加上这一堆硬币中的某个面值之和,如果这样分解之后最终的硬币数是最少的,那么问题就得到答案了。7 k/ O( d1 K3 g; Q
单是上面的文字描述太抽象,先假定以下变量:8 `" [ Y% H8 k
values[] : 保存每一种硬币的币值的数组% ^; q+ q2 k W- A. n
valueKinds :币值不同的硬币种类数量,即values[]数组的大小1 P+ W: K* S: C
money : 需要找零的面值& B7 }* c6 @# R& w N4 C
coiUsed[] : 保存面值为 i 的纸币找零所需的最小硬币数5 b& b" u+ M) h1 c5 Q! x) _* i
算法描述:
7 E# o) Z, ^ s 当求解总面值为 i 的找零最少硬币数 coiUsed[ i ] 时,将其分解成求解 coiUsed[ i – cents]和一个面值为 cents 元的硬币,由于 i – cents < i , 其解 coiUsed[ i – cents] 已经存在,如果面值为 cents 的硬币满足题意,那么最终解 coiUsed[ i ] 则等于 coiUsed[ i – cents] 再加上 1(即面值为 cents)的这一个硬币。; `: \8 c- a$ @ |4 v" r, P
下面用代码实现并测试一下:- ?" J0 A- F0 M2 O$ y
public class CoiChange {
5 C- a' ^1 ~; W- I /**
+ o. D! U8 a8 V+ c( }0 n3 ] * 硬币找零:动态规划算法4 w6 b" W; u! |2 D( s
*+ a$ v" {, a- e
* @param values
! B4 H- {7 {6 I& K% A' F * :保存每一种硬币的币值的数组2 p6 {5 i' `3 O1 t+ S8 d- [' G
* @param valueKinds
e1 R& U! i, T8 z1 M; y * :币值不同的硬币种类数量,即coinValue[]数组的大小
! D) t" B4 ?9 w5 o1 d m$ F" J * @param money
, U1 M4 H1 M5 |1 | * :需要找零的面值* `" k |2 v! `0 e9 w2 K
* @param coiUsed/ N% j4 z+ k( a7 s( w f
* :保存面值为i的纸币找零所需的最小硬币数
1 e& K' M: |. g4 v! K. K3 p& v */7 W/ d3 W, p8 t- a$ X
public static void makeChange(int[] values, int valueKinds, int money,
9 p: a9 N8 {" z7 b int[] coiUsed) {
& \+ r5 _$ t: m4 D2 ] coiUsed[0] = 0;/ f/ p! ?3 L- v
// 对每一分钱都找零,即保存子问题的解以备用,即填表
# e9 ]( d( C# A1 M' ^" w$ Z for (int cents = 1; cents <= money; cents++) {
3 U u4 X! y% c, @. M: J // 当用最小币值的硬币找零时,所需硬币数量最多# j* k$ N7 F3 d" }
int minCoi = cents;
" G7 z: \( R' I // 遍历每一种面值的硬币,看是否可作为找零的其中之一; T7 A) [* Q7 v/ o
for (int kind = 0; kind < valueKinds; kind++) {
/ E% I% v6 s; @6 I! B/ S& N! x2 l( ~ // 若当前面值的硬币小于当前的cents则分解问题并查表# J% P& l% G4 {+ n9 U
if (values[kind] <= cents) {" t; U3 Z7 a1 k$ p& g
int temp = coiUsed[cents - values[kind]] + 1;/ q) [0 v" t- B' p2 s' B
if (temp < minCoi) {( C( G- ~5 L4 j. _+ @8 W4 a& Z: m
minCoi = temp;1 ^6 X: L3 H/ x/ X+ L: }- @& \) F
}5 k/ X/ b. P6 E ] v" X: E
}( M! j" k# c! J+ N
}
5 ^. X$ B6 L: x; V% w3 Y% v, @ // 保存最小硬币数
3 i; L' U$ z6 m# Q coiUsed[cents] = minCoi;
3 a4 ~8 C/ o: A) C System.out.println("面值为 " + (cents) + " 的最小硬币数 : "
* h# ^' N4 L0 L + coiUsed[cents]);4 F- Z s8 \% j( Q1 s; }4 _
}
/ G* Y" e; U- V; x }
; g5 |% j2 Y- x- g public static void main(String[] args) {- M8 w' E- C4 i" v1 l5 ^
// 硬币面值预先已经按降序排列! J7 u! k* O; z$ x* c
int[] coinValue = new int[] { 25, 21, 10, 5, 1 };
, R7 w& |7 v5 a1 ^" F // 需要找零的面值
3 n1 g' F8 `3 B int money = 63;! G' g$ T' C A. \+ B
// 保存每一个面值找零所需的最小硬币数,0号单元舍弃不用,所以要多加1
: c" g8 V0 R5 D* s% l# J( j int[] coiUsed = new int[money + 1];
! _9 E8 z: ^- o |! m makeChange(coinValue, coinValue.length, money, coiUsed);. e5 J6 q: b% V6 I1 R& W. H
} l* Y5 [9 \- i$ S1 x! B4 Y- V
}6 |! E1 J# e1 Y3 d& j1 o& {
测试结果:9 Z4 F1 Z q# l0 | U7 W. T1 q
面值为 1 的最小硬币数 : 1 面值为 2 的最小硬币数 : 2 面值为 3 的最小硬币数 : 3 面值为 4 的最小硬币数 : 4 面值为 5 的最小硬币数 : 1 面值为 6 的最小硬币数 : 2 ... ... 面值为 60 的最小硬币数 : 3 面值为 61 的最小硬币数 : 4 面值为 62 的最小硬币数 : 4 面值为 63 的最小硬币数 : 3
' u: Q& p2 R1 l8 }2 y% j' @ 上面的代码并没有给出具体应该是哪几个面值的硬币,这个可以再使用一些数组保存而打印出来。
# n( _( b; ]# s$ b0 u' j8 ? 参考lovebeyond19831993.blog.163.com/blog/static/131811444201101901819833/
" i5 x& j7 Y8 o2 U b 问题:
& T2 @5 p7 b2 R( t 假设有n种面值不同的硬币,个个面值存于数组T[1:n]中,现在用这些硬币来找钱,各种硬币的使用个数不限;
' U/ @/ e8 W/ V. }& u 求对于给定的钱数N,最少可以由几枚硬币组成,并输出硬币序列。
5 E3 }5 H# Y$ ~9 X1 e+ g- ~ 分析:
/ a' F6 q. Y: f% Z7 H 假设对于i = 1...N-1, 所需最少的硬币数Count(i) 已知, 那么对于N,所需的硬币数为Min( Count(i) + Count(N-i)) , i=1...N-1;
: t, W* O# l# t 于是一个直观的方法是用递归计算。
; u9 H( h4 t" T- N" U( U 但是,递归过程中,每次计算Count(i),都会重复计算 Count(1)....Count(i-1); 这样时间复杂度就是O(N2);) o3 C' n& i7 M6 ?! L( p: i# a, i5 e6 x; W
事实上,这是一个典型的动态规划问题,我们可以从1开始记录下每个钱数所需的硬币枚数,避免重复计算,为了能够输出硬币序列,我们还需要记录下每次新加入的硬币。
! a6 C" M% R: G( `+ ` 下面给出用动态规划解决此问题的递推式:1 u1 ~! I& ]9 k' u. @( h7 n# j5 L% E
参数说明: 当只用面值为T[1],T[2],…T[n]来找出钱j时,所用的硬币的最小个数记为C(i,j),则C(i,j)的递推方程为:
& l. U5 ]6 E! A* e- J) {2 d 算法设计:
; I4 N9 y( [$ `% B+ _3 j #include <iostream>' g$ h: Y- G2 [2 H6 V3 b
using namespace std;
% ~: [" `; m: C4 L1 ~* A7 _! u! h int min_number_of_stamps(cot int* array, size_t array_size, int request) {
# O% d; C) F) Q8 h2 c if (array == NULL || array_size <= 0 || request < 1) {9 o& l/ Y( L8 ]& E9 S! L* u
return -1;: Q; G, w$ d! e: a' ?2 H% a
}* w7 a& V+ O% j' k; u
int* numofStamps = new int[request + 1];) Q' g: k Z* Q% R
numofStamps[0] = 0;3 |1 s: _" T% s7 n5 z9 c
for(int i = 1; i <= request; i++){
& b0 ]" j# E' ?* Z int cn = i;' U% o& k# _4 L
for(size_t j = 0; j < array_size; j++){" y+ O4 M+ p; X+ ^, ^
if(array[j] <= i){6 w' m) Z l9 a: d
int tmp = numofStamps[i - array[j]] + 1;" o- F2 t# ]: H. u8 w
if(tmp < cn)
+ a& R* t; M: z$ |7 Q cn = tmp;/ J$ h3 d; C! L4 K3 z8 z0 W0 q
}4 e; B. F, D- j; ?
} L4 w4 W: f, z# E7 b
numofStamps[i] = cn;& Z0 a7 ^. n9 n8 q4 ?2 m' z* X2 u% o
}
5 w( S. C7 |4 q X* s- Z H int t = numofStamps[request];$ R1 ]) g0 E" C' A: a
delete[] numofStamps;
' l1 |0 X, X0 x; J4 [ return t;
5 w6 r/ P( V# b6 v8 V }
; z! M' ~, ~! s6 y/ }( b, j int main() {+ j1 u: G# o k! ~" n& [# G3 {! Q3 A" A4 H
int array[] = { 90, 30, 24, 15, 12, 10, 5, 3, 2, 1 };
6 x& i7 o+ _7 b/ T- C; G( L. P int array_size = sizeof(array) / sizeof(int);2 w C% c% w4 }. F& V' ~- n, J: ~
int t = min_number_of_stamps(array, array_size, 32);
' {3 p2 j+ P4 i) u9 l: M, ]; r cout << t << endl;
( b7 R2 D9 j1 _2 B" D* p }
, D$ B$ V/ y4 Y' b1 f1 J" L# Q. W& q |
|