目标和
Tips
题目类型: Dynamic Programming
题目
给定一个非负整数数组, a1
, a2
, ..., an
, 和一个目标数 S
. 现在你有两个符号 +
和 -
. 对于数组中的任意一个整数, 你都可以从 +
或 -
中选择一个符号添加在前面. 返回可以使最终数组和为目标数 S
的所有添加符号的方法数.
示例
输入: nums: [1, 1, 1, 1, 1], S: 3
输出: 5
解释:
- -1+1+1+1+1 = 3
- +1-1+1+1+1 = 3
- +1+1-1+1+1 = 3
- +1+1+1-1+1 = 3
- +1+1+1+1-1 = 3
一共有 5 种方法让最终目标和为 3.
题解
0-1 背包问题是选还是不选, 需要先将这个题目转化一下. 设 x
为所有前面为 +
的元素之和, 设 y
为所有前面为 -
的元素之和的绝对值, 那么记录一个总量 sum = x + y
,
并且 S = x - y
, 因此 S + sum = 2 * x
, 因此 x = (S + sum) / 2
, 即装满容量为 (S + sum) / 2
的背包, 有几种方案.
- JavaScript - 二维数组
- JavaScript - 一维数组
- Rust
/**
* @param {number[]} nums
* @param {number} target
* @return {number}
*/
var findTargetSumWays = function (nums, target) {
const n = nums.length
const sum = nums.reduce((acc, val) => acc + val)
if (Math.abs(target) > sum || (sum + target) % 2 === 1) return 0
const capacity = (target + sum) / 2
const dp = new Array(n + 1).fill(0).map(() => new Array(capacity + 1).fill(0))
dp[0][0] = 1
for (let i = 1; i <= n; i++) {
for (let w = 0; w <= capacity; w++) {
if (w - nums[i - 1] < 0) {
dp[i][w] = dp[i - 1][w]
} else {
dp[i][w] = dp[i - 1][w] + dp[i - 1][w - nums[i - 1]]
}
}
}
return dp[n][capacity]
}
/**
* @param {number[]} nums
* @param {number} target
* @return {number}
*/
var findTargetSumWays = function (nums, target) {
const n = nums.length
const sum = nums.reduce((acc, val) => acc + val)
if (Math.abs(target) > sum || (sum + target) % 2 === 1) return 0
const capacity = (sum + target) / 2
const dp = new Array(capacity + 1).fill(0)
dp[0] = 1
for (let i = 0; i < n; i++) {
for (let j = capacity; j >= nums[i]; j--) {
dp[j] = dp[j] + dp[j - nums[i]]
}
}
return dp[capacity]
}
pub fn find_target_sum_ways(nums: Vec<i32>, target: i32) -> i32 {
let n = nums.len();
let sum = nums.iter().sum();
if (target.abs() > sum || (sum + target) % 2 == 1) {
return 0;
}
let capacity = (sum + target) / 2;
let mut dp = vec![0; (capacity + 1) as usize];
dp[0] = 1;
for i in 1..=n {
for j in (nums[i - 1]..=capacity).rev() {
dp[j as usize] = dp[j as usize] + dp[(j - nums[i - 1]) as usize];
}
}
dp[capacity as usize]
}