当前位置:   article > 正文

背包问题(416,494,474,322,518,139)_python给你一个 只包含正整数 的 非空 数组 nums 。请你判断是否可以将这个数组分

python给你一个 只包含正整数 的 非空 数组 nums 。请你判断是否可以将这个数组分

 一、416. 分割等和子集

1.1 题目描述

给你一个 只包含正整数 的 非空 数组 nums 。请你判断是否可以将这个数组分割成两个子集,使得两个子集的元素和相等。

1.2 代码

题解:https://leetcode-cn.com/problems/partition-equal-subset-sum/solution/fen-ge-deng-he-zi-ji-by-leetcode-solution/

力扣

创建二维数组dp,包含 n 行,target+1 列。其中dp[i] [j] 表示从数组的[0,i] 下标范围内选取若干个正整数(可以是 0 个),是否存在一种选取方案使得被选取的正整数的和等于 j。初始时,dp 中的全部元素都是 false。

  1. public boolean canPartition(int[] nums) {
  2. int len = nums.length;
  3. int sum=0;
  4. for(int num:nums) {
  5. sum+=num;
  6. }
  7. if(sum%2!=0) return false;
  8. int target=sum/2;
  9. boolean dp[][]=new boolean[len][target+1];
  10. dp[0][0] = true;
  11. if (nums[0]<=target) dp[0][nums[0]]=true;
  12. for (int i = 1; i < len; i++) {
  13. for(int j=0;j <=target;j++) {
  14. dp[i][j] = dp[i - 1][j];
  15. if (nums[i] <= j) {
  16. dp[i][j]=dp[i-1][j] || dp[i-1][j-nums[i]];
  17. }
  18. }
  19. if (dp[i][target]) {//提前结束
  20. return true;
  21. }
  22. }
  23. return dp[len-1][target];
  24. }

494. 目标和

 

  1. class Solution494 {
  2. int count=0;
  3. public int findTargetSumWays(int[] nums, int target) {
  4. backing(nums,target,0,0);
  5. return count;
  6. }
  7. public void backing(int[] nums,int target,int curSum,int index){
  8. if (index==nums.length) {
  9. if (curSum==target){
  10. count++;
  11. return;
  12. }
  13. }else {
  14. backing(nums,target,curSum+nums[index],index+1);
  15. backing(nums,target,curSum-nums[index],index+1);
  16. }
  17. }
  18. }

474. 一和零

题解

dp[i][j][k] 表示输入字符串在子区间 [0, i] 能够使用 j 个 0 和 k 个 1 的字符串的最大数量。

  1. /**
  2. * 474. 一和零
  3. * dp[i][j][k] 表示输入字符串在子区间 [0, i] 能够使用 j 个 0 和 k 个 1 的字符串的最大数量。
  4. */
  5. class Solution {
  6. public int findMaxForm(String[] strs, int m, int n) {
  7. int len=strs.length;
  8. int[][][] dp = new int[len+1][m+1][n+1];
  9. for (int i = 1; i <= len; i++) {
  10. int[] count=countZeroOne(strs[i-1]);
  11. for (int j = 0; j <= m; j++) {
  12. for (int k = 0; k <= n; k++) {
  13. dp[i][j][k] = dp[i - 1][j][k];
  14. if (count[0]<=j && count[1]<=k){
  15. dp[i][j][k]=Math.max(dp[i-1][j][k],dp[i-1][j-count[0]][k-count[1]]+1);
  16. }
  17. }
  18. }
  19. }
  20. return dp[len][m][n];
  21. }
  22. public int[] countZeroOne(String s){
  23. int[] count = new int[2];
  24. for (char ch:s.toCharArray()){
  25. count[ch-'0']++;
  26. }
  27. return count;
  28. }
  29. }

 322. 零钱兑换

  1. public int coinChange(int[] coins, int amount) {
  2. int[] dp = new int[amount + 1];
  3. Arrays.sort(coins);
  4. Arrays.fill(dp,amount+1);
  5. dp[0]=0;
  6. for (int i = 1; i <= amount; i++) {
  7. for (int coin : coins) {
  8. if (i >= coin) {
  9. dp[i] = Math.min(dp[i], dp[i - coin]+1);
  10. }else break;
  11. }
  12. }
  13. return dp[amount]>amount?-1:dp[amount];
  14. }

518. 零钱兑换 II

题解 

注意,此题是组合数,不是排列数,(1,2)和(2,1)视为一种,因此注意不要重复计算。 

  1. /**
  2. * 518. 零钱兑换 II,有几种方式可以凑成总金额
  3. * 外层循环是coin,因此先排列金额1,再排列金额2的,不会重复
  4. * 例如dp[3]
  5. * 3=1+1+1;
  6. * 3=1+2;不会重复
  7. */
  8. public int change(int amount, int[] coins) {
  9. int[] dp=new int[amount+1];
  10. dp[0]=1;
  11. for (int coin:coins){
  12. for (int i = coin; i <=amount ; i++) {
  13. dp[i]+=dp[i-coin];
  14. }
  15. }
  16. return dp[amount];
  17. }

139. 单词拆分

  1. public boolean wordBreak(String s, List<String> wordDict) {
  2. int len=s.length();
  3. boolean[] dp=new boolean[len+1];
  4. dp[0]=true;//包啥都不装
  5. for (int i = 1; i < len+1; i++) {//i代表现在遍历到的长度
  6. for(String word:wordDict){
  7. int wordLen=word.length();//物品的长度
  8. if (i>=wordLen && word.equals(s.substring(i-wordLen,i))){
  9. //背包大小>=物品大小
  10. dp[i]=dp[i]||dp[i-wordLen];//选择装包还是不装(装的前提是前i-len个元素匹配上了)
  11. }
  12. }
  13. }
  14. return dp[len];
  15. }

 

声明:本文内容由网友自发贡献,转载请注明出处:【wpsshop博客】
推荐阅读
相关标签
  

闽ICP备14008679号