代码随想录第三十七天(完全背包问题)|完全背包|零钱兑换 II|组合总和 Ⅳ

本文主要是介绍代码随想录第三十七天(完全背包问题)|完全背包|零钱兑换 II|组合总和 Ⅳ,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

完全背包问题

完全背包问题和01背包的区别就在于每一个物品可取的次数,01背包每个物品只能取一次,完全背包每个物品能取无数次。

而01背包为了保证每个物品只取一次,在遍历背包的时候需要倒序遍历,这样才能保证之前的状态都是初始状态,没有添加过物品,利用之前的状态时就不会将同一个物品进行重复添加。

在完全背包中,就需要每个物品能取多次,于是解除遍历背包时的倒序限制就行。

import java.util.*;
public class Main{public static void main (String[] args) {Scanner sc=new Scanner(System.in);int M=sc.nextInt();int N=sc.nextInt();int[] weight=new int[M];int[] value=new int[M];int bagsize=N;for(int i=0;i<M;i++){weight[i]=sc.nextInt();value[i]=sc.nextInt();}maxValue(weight,value,bagsize);}public static void maxValue(int[] weight,int[] value,int bagsize){int[] dp=new int[bagsize+1];for(int i=0;i<weight.length;i++){for(int j=weight[i];j<bagsize+1;j++){dp[j]=Math.max(dp[j],dp[j-weight[i]]+value[i]);}}System.out.println(dp[bagsize]);}
}

在完全背包中,在求装满指定容量背包的组合总数的时候,遍历顺序非常重要,只求组合总数的话,先遍历物品,后遍历背包,因为先遍历物品的话,每次都会将前一个物品先添加进去,只会出现{1,2}这样的组合,不会出现{2,1}的组合。因为当前物品层依赖上一个物品层传递下来的状态,上一个物品层,就只会又上一个物品层和其之前的物品层的组合。不会有之后物品层的值放入组合中。

而将遍历顺序颠倒,先遍历背包,后遍历物品,以物品{1,2,3}为例,装满容量为4的背包,y轴为物品,x轴为背包。

01234
011124
111236
211247

当背包容量为3,假如已取物品1,也就是先放入重量为2的物品,剩余只能取重量为1的组合了,就出现了{2,1},而假如已取的是物品0,也就是先放入重量为1的物品,剩余只能取重量为2的组合,因为在上一个状态遍历了物品1,已经有了重量2的组合,就出现了{1,2}和{1,1,1}的组合了。

零钱兑换 II

题中只要求求组合总数,不讲究顺序。

动规五部曲,改变下遍历顺序

class Solution {public int change(int amount, int[] coins) {int[] dp=new int[amount+1];dp[0]=1;for(int i=0;i<coins.length;i++){for(int j=coins[i];j<amount+1;j++){dp[j]+=dp[j-coins[i]];}}return dp[amount];}
}

组合总和 Ⅳ

讲究顺序

class Solution {public int combinationSum4(int[] nums, int target) {int[] dp=new int[target+1];dp[0]=1;for(int j=1;j<target+1;j++){for(int i=0;i<nums.length;i++){if(j>=nums[i]){dp[j]+=dp[j-nums[i]];}}}return dp[target];}
}

这篇关于代码随想录第三十七天(完全背包问题)|完全背包|零钱兑换 II|组合总和 Ⅳ的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/965506

相关文章

好题——hdu2522(小数问题:求1/n的第一个循环节)

好喜欢这题,第一次做小数问题,一开始真心没思路,然后参考了网上的一些资料。 知识点***********************************无限不循环小数即无理数,不能写作两整数之比*****************************(一开始没想到,小学没学好) 此题1/n肯定是一个有限循环小数,了解这些后就能做此题了。 按照除法的机制,用一个函数表示出来就可以了,代码如下

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

poj2576(二维背包)

题意:n个人分成两组,两组人数只差小于1 , 并且体重只差最小 对于人数要求恰好装满,对于体重要求尽量多,一开始没做出来,看了下解题,按照自己的感觉写,然后a了 状态转移方程:dp[i][j] = max(dp[i][j],dp[i-1][j-c[k]]+c[k]);其中i表示人数,j表示背包容量,k表示输入的体重的 代码如下: #include<iostream>#include<

hdu2159(二维背包)

这是我的第一道二维背包题,没想到自己一下子就A了,但是代码写的比较乱,下面的代码是我有重新修改的 状态转移:dp[i][j] = max(dp[i][j], dp[i-1][j-c[z]]+v[z]); 其中dp[i][j]表示,打了i个怪物,消耗j的耐力值,所得到的最大经验值 代码如下: #include<iostream>#include<algorithm>#include<

csu(背包的变形题)

题目链接 这是一道背包的变形题目。好题呀 题意:给n个怪物,m个人,每个人的魔法消耗和魔法伤害不同,求打死所有怪物所需的魔法 #include<iostream>#include<algorithm>#include<cstring>#include<stack>#include<queue>#include<set>//#include<u>#include<map

hdu1011(背包树形DP)

没有完全理解这题, m个人,攻打一个map,map的入口是1,在攻打某个结点之前要先攻打其他一个结点 dp[i][j]表示m个人攻打以第i个结点为根节点的子树得到的最优解 状态转移dp[i][ j ] = max(dp[i][j], dp[i][k]+dp[t][j-k]),其中t是i结点的子节点 代码如下: #include<iostream>#include<algorithm

hdu4869(逆元+求组合数)

//输入n,m,n表示翻牌的次数,m表示牌的数目,求经过n次操作后共有几种状态#include<iostream>#include<algorithm>#include<cstring>#include<stack>#include<queue>#include<set>#include<map>#include<stdio.h>#include<stdlib.h>#includ

hdu1171(母函数或多重背包)

题意:把物品分成两份,使得价值最接近 可以用背包,或者是母函数来解,母函数(1 + x^v+x^2v+.....+x^num*v)(1 + x^v+x^2v+.....+x^num*v)(1 + x^v+x^2v+.....+x^num*v) 其中指数为价值,每一项的数目为(该物品数+1)个 代码如下: #include<iostream>#include<algorithm>

活用c4d官方开发文档查询代码

当你问AI助手比如豆包,如何用python禁止掉xpresso标签时候,它会提示到 这时候要用到两个东西。https://developers.maxon.net/论坛搜索和开发文档 比如这里我就在官方找到正确的id描述 然后我就把参数标签换过来

购买磨轮平衡机时应该注意什么问题和技巧

在购买磨轮平衡机时,您应该注意以下几个关键点: 平衡精度 平衡精度是衡量平衡机性能的核心指标,直接影响到不平衡量的检测与校准的准确性,从而决定磨轮的振动和噪声水平。高精度的平衡机能显著减少振动和噪声,提高磨削加工的精度。 转速范围 宽广的转速范围意味着平衡机能够处理更多种类的磨轮,适应不同的工作条件和规格要求。 振动监测能力 振动监测能力是评估平衡机性能的重要因素。通过传感器实时监