DP优化与换零钱问题

1 当贪心不再起效的时候

  对于换零钱问题,最简单也是性能最好的方法就是贪心算法。可是贪心算法一定要满足面值相邻两个零钱至少为二倍关系的前提条件。例如1,2,5,10,20……这样的零钱组应用贪心最简单;可对于1,3,4,5,6,10……这样的零钱组就不起效了:当目标总值是12的时候,应用贪心算法:答案是10x1+1x2,共3张,可正确答案应该是6x2共两张。

  那怎么解决呢?这样的问题符合动态规划的特点:任何一个状态可以由前边的状态计算而得。

  定义问题(明确需求):一组不重复且升序排列零钱面额数组vector<int> c(共m个元素),目标面值:int n,求所需最少的零钱张数。

2 O(mn)空间复杂度DP算法

  就像背包问题。最经典的算法是生成一个(m+1)x(n+1)DP数组来保存中间值,其中dp[i][j]表示只利用第1~i种零钱凑成目标面值为j所需的最少零钱张数。

  我们很容易得出状态转移方程:

    dp[i][j]=min(0+dp[i-1][j],1+dp[i-1][j-1*c[i]],2+dp[i-1][j-2*c[i]], ······);

  从而很容易写出代码。

  但作为一个追(xian)求(de)卓(dan)越(teng)的程序员,我又怎么会满足于此?看起来时间复杂度O(mn)已经达到极至,可空间复杂度O(mn)好像可以继续优化!

3 O(n)空间复杂度DP算法

  由上边的状态转移方程,我们发现每个元素只用到了上一行的若干个元素。那其实我们很容易想到,一个大小为2*(n+1)的一维数组就足够了!

  状态转移方程:

    dp[index][j]=min(0+dp[1-index][j],1+dp[1-index][j-1*c[i]],2+dp[1-index][j-2*c[i]], ······);

  计算方法没有任何差别,但是空间复杂度却大大降低!

  不仅如此,对于有一些DP算法,每个元素的计算只需要用到当前行左边的元素和上一行相同列坐标元素的值(例如矩阵的最小最径的DP算法),这样的算法,我们可以直接用一个n+1的一维数组作为DP数组就够用了!

4 总结

  由此,我们可以总结出本文的中心思想:

  (1)一个二维DP算法,如果每个元素的计算只需要用到上一行的若干个元素的值的时候,我们都可以用两个一维DP数组来优化空间复杂度。

  (2)一个二维DP算法,如果每个元素的计算只需要用到当前行左边的元素和上一行相同列坐标元素的值的时候,我们都可以用一个一维DP数组来优化空间复杂度。

  (3)要多尝试降低二维DP算法的空间复杂度。

  鼓掌[逃]!

转载于:https://www.cnblogs.com/qiaoconglovelife/p/5815966.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值