题解 | #兑换零钱(一)#
兑换零钱(一)
https://www.nowcoder.com/practice/3911a20b3f8743058214ceaa099eeb45
dp[j]表示兑换成j元最少需要多少张纸币,dp[0] = 0。 dp[j] = min( dp[j],dp[j-arr[i]] + 1 ) 转移方程表示兑换成j元要么用i-1张货币直接兑换成j元,要么i-1张货币兑换成j-arr[i]元,再加一张价值arr[i]的纸币。 这里的转移方程是对原本二维的dp[i][j]压缩纬度之后的结果,原本dp[i][j]表示用i张货币兑换成j元最少用多少张。 原本对于i的循环还在。因为在i=1的情况下对j进行遍历填充dp表之后,i+1之后dp[j]中保存的是i=1的情况,就是之前的状态,更新最新状态的时候正好需要,所以对dp表进行压缩。 注意边界条件,j要从arr[i]的数值开始,因为用价值arr[i]的货币去兑换小于其本身价值的货币是没有意义的 int min( const int a, const int b ) { return a<b?a:b; } int minMoney(int* arr, int arrLen, int aim ) { int dp[aim+1]; memset( dp,100,sizeof(dp) ); //初始化dp表,初始化dp表后,表内的数据是很大的,并不是100 dp[0] = 0; for( int i =0; i < arrLen; ++i ) { for( int j = arr[i]; j <= aim; ++j ) { dp[j] = min( dp[j],dp[j-arr[i]] + 1 ); } } return dp[aim]>aim?-1:dp[aim]; //初始化dp表的数值是很大的,如果dp表的某个值大于aim表示其根本没有更新,也就是无法兑换 }