您的位置:首页 > 游戏 > 手游 > 力扣题解(零钱兑换)

力扣题解(零钱兑换)

2024/10/7 2:25:50 来源:https://blog.csdn.net/yyssas/article/details/140568941  浏览:    关键词:力扣题解(零钱兑换)

322. 零钱兑换

给你一个整数数组 coins ,表示不同面额的硬币;以及一个整数 amount ,表示总金额。

计算并返回可以凑成总金额所需的 最少的硬币个数 。如果没有任何一种硬币组合能组成总金额,返回 -1 。

你可以认为每种硬币的数量是无限的。

思路:本题可以看成是对一堆给定的数字,选取其中某几个,组合的结果是amount,因此是完全背包问题。

dp[i][j]表示前i个物品,结果是j所需的最小银币个数。

dp[i][j]=min(dp[i-1][j],dp[i][j-coins[i]]),此处dp[i-1][j]表示不要第i个数,dp[i][j-coins[i]]表示至少要一个第i个数,因此二者会把所有可能包含在内。

初始化:

j为0的时候,一定有且仅有一种组成方式,就是一个数字都不放进去。而i为0时,除了j为0的情况,其余情况均不可能实现,因此可以初始化为一个很大的数,因为dp[i][j]求得是最小值,因此作为一个很大的数,一定不会是dp[i][j]的构成,即隐含了将不存在的情况排除在外的判断那一步。

class Solution {
public:int coinChange(vector<int>& coins, int amount) {int n=coins.size();int INF=0x3f3f3f3f;vector<vector<int>>dp(n+1,vector<int>(amount+1,INF));for(int i=0;i<=n;i++){dp[i][0]=0;}coins.insert(coins.begin(),0);for(int i=1;i<=n;i++){for(int j=1;j<=amount;j++){dp[i][j]=dp[i-1][j];if(j-coins[i]>=0){//  cout<<i<<j-coins[i]<<" ||";dp[i][j]=min(dp[i][j],dp[i][j-coins[i]]+1);}}}return dp[n][amount]==INF?-1:dp[n][amount];}
};

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com