494. 目标和

该博客讨论了如何利用动态规划解决LeetCode上的‘目标和’问题,这是一个与01背包问题相关的组合问题。文章指出,当目标和与数组元素之和的奇偶性不匹配或目标和超过数组元素之和时,没有解决方案。通过设置dp数组并用动态规划的方法,博主展示了如何计算可以构造不同表达式的数目,以达到特定的目标和。
摘要由CSDN通过智能技术生成

给你一个整数数组 nums 和一个整数 target 。

向数组中的每个整数前添加 ‘+’ 或 ‘-’ ,然后串联起所有整数,可以构造一个 表达式 :

例如,nums = [2, 1] ,可以在 2 之前添加 ‘+’ ,在 1 之前添加 ‘-’ ,然后串联起来得到表达式 “+2-1” 。
返回可以通过上述方法构造的、运算结果等于 target 的不同 表达式 的数目。

来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/target-sum
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。
原作者:https://leetcode-cn.com/problems/target-sum/solution/dai-ma-sui-xiang-lu-494-mu-biao-he-01bei-rte9/
01背包问题,问题转化为装满容量为x背包,有几种方法。
两种特判,if ((S + sum) % 2 == 1) return 0; // 此时没有方案。if (abs(S) > sum) return 0; // 此时没有方案。
再回归到01背包问题,为什么是01背包呢?

因为每个物品(题目中的1)只用一次!

这次和之前遇到的背包问题不一样了,之前都是求容量为j的背包,最多能装多少。

本题则是装满有几种方法。其实这就是一个组合问题了。
把 所有的 dp[j - nums[i]] 累加起来。
所以求组合类问题的公式,都是类似这种:
dp[j] += dp[j - nums[i]]
从递归公式可以看出,在初始化的时候dp[0] 一定要初始化为1,因为dp[0]是在公式中一切递推结果的起源,如果dp[0]是0的话,递归结果将都是0。

dp[0] = 1,理论上也很好解释,装满容量为0的背包,有1种方法,就是装0件物品。
是0的话,递归结果将都是0。

dp[0] = 1,理论上也很好解释,装满容量为0的背包,有1种方法,就是装0件物品。

dp[j]其他下标对应的数值应该初始化为0,从递归公式也可以看出,dp[j]要保证是0的初始值,才能正确的由dp[j - nums[i]]推导出来。

class Solution {
public:
    int findTargetSumWays(vector<int>& nums, int target) {
        int dp[1005],sum=0,res=0;
        memset(dp,0,sizeof(dp));
        dp[0]=1;
        for(int i=0;i<nums.size();i++)
            sum+=nums[i];
        if(abs(target)>sum) return 0;
        int n=(sum+target)/2;
        if((sum+target)%2!=0) return 0;
        cout<<sum<<" "<<target<<endl;
        for(int i=0;i<nums.size();i++)
            {
                for(int j=n;j>=nums[i];j--)
                    dp[j]+=dp[j-nums[i]];
            }
        return dp[n];
    }
};
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值