dp专题15 零钱兑换

本题链接:力扣(LeetCode)官网 - 全球极客挚爱的技术成长平台

题目:

思路:        

        这道题,是个比较模板的完全背包问题,这里要求的是问凑成总金额所需的最少的硬币的个数。

我们明确一下 dp[ i ] 的含义,是   dp[ i ] 存储的是凑成总金额所需的最少的硬币的个数。

其中 下标 i 的含义是  "背包容量" amount ,  并且不同金额中是有无限个的。

所以我们可以将,不同金额硬币作为"物品体积",每个硬币价值都为 1 表示物品的数量。

其次我们也要分析 dp 数组的初始化,由于 要求所需最少的硬币个数,所以我们将dp初始化全部为正无穷。 其中 dp[ 0 ] 要从初始化为 0  ,毕竟 凑成总金额为 0 所需的硬币个数就是 0.

随后确定 dp 公式:    dp[ j ] = min(dp[ j ] , dp[ j - i ] + 1 );

最后我们选取的过程中,选取顺序无关,所以是个组合数,确定遍历顺序。

先物品,后背包。

代码详解如下:

class Solution {
public:
    const int INF = 0x3f3f3f3f3f3f3f;   // 定义正无穷
    inline int coinChange(vector<int>& coins, int &amount) 
{
	vector<int>dp(amount + 2,INF);  // 开辟 dp 数组,并初始化为 INF
	dp[0] = 0;  // 特定初始化
	
    // 组合数的遍历物品以及背包
	for(int &i:coins)   
	{
		for(int j = i;j <= amount;++j)  // 完全背包正向遍历
		{
			dp[j] = min(dp[j],dp[j - i] + 1);   // dp 公式
		}
	}
	if(dp[amount] >= INF) return -1;    // 返回结果
	else return dp[amount];
}
};

 最后提交:

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值