完全背包问题

完全背包问题与01背包问题基本相同,就是倒序改为正序,正序会重复。

lk518 518. 零钱兑换 II - 力扣(LeetCode)

class Solution {
public:
typedef long long ll;
    ll change(int amount, vector<int>& coins) {
        vector<ll> dp(amount+1,0);
        dp[0]=1;
        for(int i=0;i<coins.size();i++){
            for(int j=coins[i];j<=amount;j++){
                if (dp[j] < INT_MAX - dp[j - coins[i]])//用相减是因为相加会溢出
                dp[j]+=dp[j-coins[i]];
            }
        }
        return dp[amount];
    }
};

这里不写dp[j]+dp[j-coins[i]]是因为两者先直接加起来可能会越界,所以写成以上形式。

lk 377 377. 组合总和 Ⅳ - 力扣(LeetCode)

class Solution {
public:
    int combinationSum4(vector<int>& nums, int target) {

        vector<int> dp(target+1,0);
        dp[0]=1;
        for(int j=0;j<=target;j++){
            for(int i=0;i<nums.size();i++){
                if (j >= nums[i] && dp[j] < INT_MAX -dp[j - nums[i]])//为什么用相减,因为相加会溢出
                dp[j]+=dp[j-nums[i]];
            }
        }
        return dp[target];
    }
};

这里题目相当于是求排列数,排列数就是可以重复的,所以先遍历背包j后遍历物品i,每次遍历背包都会重复遍历物品从1开始,所以会重复,这里因为i在里层,所以外层j不能从nums【i】开始,必须在内曾增加一个条件,if(j>=nums[i])才行;

lk 322 322. 零钱兑换 - 力扣(LeetCode)

class Solution {
public:
    int coinChange(vector<int>& coins, int amount) {
        vector<int> dp(amount+1,INT_MAX);
        dp[0]=0;
        for(int i=0;i<coins.size();i++){
            for(int j=coins[i];j<=amount;j++){
                if(dp[j-coins[i]]!=INT_MAX)
               dp[j] = min(dp[j - coins[i]] + 1, dp[j]);
            }
        }
        if(dp[amount]==INT_MAX) return -1;
        return dp[amount];
    }
};

这题要求要是组合数,就是不能重复的,所以先便利物品在遍历背包,物品从1,2开始,所以不会重复,这里要加条件dp[j-coins[i]]<INT_MAX或者如题目所示不等于也行,因为后面dp[j - coins[i]] + 1可能会越界。

lk 279 279. 完全平方数 - 力扣(LeetCode)

class Solution {
public:
    int numSquares(int n) {
        vector<int> dp(n+1,INT_MAX);
        dp[0]=0;
        dp[1]=1;
        for(int i=1;i<=n/i;i++){
            for(int j=i*i;j<=n;j++){
                if(dp[j-i*i]!=INT_MAX)
                dp[j]=min(dp[j-i*i]+1,dp[j]);
            }
        }
        return dp[n];
    }
};

这里是要求组合数,要求不能重复,因为没有数组,所有i直接从1开始,同样,写if(dp[j-i*i]<INT_MAX)也可以,完全平方数的最小数量所以要用min把所有dp数组初始化为最大。这里min要和dp【i】自己取最小值,因为12可能是由3个4组成,也可能是由4个3组成,要找出符合相加等于12的最小个数。

lk 139 139. 单词拆分 - 力扣(LeetCode)

class Solution {
public:
    bool wordBreak(string s, vector<string>& wordDict) {
        unordered_set<string> wordset(wordDict.begin(),wordDict.end());
        vector<bool> dp(s.size()+1,false);
        dp[0]=true;
        for(int j=1;j<=s.size();j++){
            for(int i=0;i<j;i++){
                string word=s.substr(i,j-i);
                if(wordset.find(word)!=wordset.end()&&dp[i])
                dp[j]=true;
            }
        }
        return dp[s.size()];
    }
};

要用wordset.find()找word

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值