1049. 最后一块石头的重量II
本题其实就是尽量让石头分成重量相同的两堆,相撞之后剩下的石头最小,这样就化解成01背包问题了。
五部曲:
- dp[j]表示容量(这里说容量更形象,其实就是重量)为j的背包,最多可以背最大重量为dp[j]。
- dp[j] = max(dp[j], dp[j - stones[i]] + stones[i])
- 我们要求的target其实只是最大重量的一半,所以dp数组开到15000大小就可以了。
- 如果使用一维dp数组,物品遍历的for循环放在外层,遍历背包的for循环放在内层,且内层for循环倒序遍历!
- 最后dp[target]里是容量为target的背包所能背的最大重量。那么分成两堆石头,一堆石头的总重量是dp[target],另一堆就是sum - dp[target]。在计算target的时候,target = sum / 2 因为是向下取整,所以sum - dp[target] 一定是大于等于dp[target]的。
那么相撞之后剩下的最小石头重量就是 (sum - dp[target]) - dp[target]。
/**
* @param {number[]} stones
* @return {number}
*/
var lastStoneWeightII = function (stones) {
let sum = stones.reduce((s, n) => s + n)
let dpLen = Math.floor(sum / 2)
let dp = new Array(dpLen + 1).fill(0)
for (let i = 0; i < stones.length; i++) {
for (let j = dpLen; j >= stones[i]; j--) {
dp[j] = Math.max(dp[j], dp[j - stones[i]] + stones[i])
}
}
return sum - dp[dpLen] - dp[dpLen]
};
494. 目标和
详见leetcode官方题解(二维数组更清晰)
/**
* @param {number[]} nums
* @param {number} target
* @return {number}
*/
var findTargetSumWays = function (nums, target) {
let sum = 0;
for (const num of nums) {
sum += num;
}
const diff = sum - target;
if (diff < 0 || diff % 2 !== 0) {
return 0;
}
const n = nums.length, neg = diff / 2;
const dp = new Array(n + 1).fill(0).map(() => new Array(neg + 1).fill(0));
dp[0][0] = 1;
for (let i = 1; i <= n; i++) {
const num = nums[i - 1];
for (let j = 0; j <= neg; j++) {
dp[i][j] = dp[i - 1][j];
if (j >= num) {
dp[i][j] += dp[i - 1][j - num];
}
}
}
return dp[n][neg];
};
474. 一和零
详见leetcode官方题解(01背包问题,需要三维数组)
/**
* @param {string[]} strs
* @param {number} m
* @param {number} n
* @return {number}
*/
var findMaxForm = function (strs, m, n) {
const length = strs.length;
const dp = new Array(length + 1).fill(0).map(() => new Array(m + 1).fill(0).map(() => new Array(n + 1).fill(0)));
for (let i = 1; i <= length; i++) {
const zerosOnes = getZerosOnes(strs[i - 1]);
let zeros = zerosOnes[0], ones = zerosOnes[1];
for (let j = 0; j <= m; j++) {
for (let k = 0; k <= n; k++) {
dp[i][j][k] = dp[i - 1][j][k];
if (j >= zeros && k >= ones) {
dp[i][j][k] = Math.max(dp[i][j][k], dp[i - 1][j - zeros][k - ones] + 1);
}
}
}
}
return dp[length][m][n];
};
const getZerosOnes = (str) => {
const zerosOnes = new Array(2).fill(0);
const length = str.length;
for (let i = 0; i < length; i++) {
zerosOnes[str[i].charCodeAt() - '0'.charCodeAt()]++;
}
return zerosOnes;
}