- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
java解法
有一个容量上限为 maxPower 的背包,以及 count 个物品,每个物品的“能力值”存储在数组 powerOutputs 中。
每个物品只能被选择一次,目标是找到一个组合,使得总能力值尽可能接近(但不超过)maxPower,并输出最大总和。
解决方案
这是一个典型的 0-1 背包问题:
使用动态规划解决,定义 dp[j] 表示在容量为 j 时能够达到的最大能力值。
遍历每个物品,对于每个物品的能力值 powerOutputs[i],更新 dp 数组:
如果选择当前物品,更新 dp[j] = max(dp[j], dp[j - powerOutputs[i]] + powerOutputs[i])。
如果不选择当前物品,保持 dp[j] 不变。
使用倒序更新 dp 数组以确保每个物品只被使用一次。
算法流程
初始化一个大小为 maxPower + 1 的数组 dp,所有元素初始化为 0。
遍历每个物品的能力值 powerOutputs[i]:
从后向前遍历容量 j,范围是 [maxPower, powerOutputs[i]]。
更新 dp[j] 的值。
遍历所有物品后,dp[maxPower] 即为能够达到的最大总能力值
import java.util.Scanner;
public class Main {
public static void main(String[] args) {
Scanner input = new Scanner(System.in);
int count = input.nextInt();
int[] powerOutputs = new int[count];
for (int i = 0; i < count; i++) {
powerOutputs[i] = input.nextInt();
}
int maxPower = input.nextInt();
System.out.println(findOptimalPower(count, powerOutputs, maxPower));
}
public static int findOptimalPower(int count, int[] powerOutputs, int maxPower) {
int[] dp = new int[maxPower + 1];
for (int i = 0; i < count; i++) {
for (int j = maxPower; j >= powerOutputs[i]; j--) {
dp[j] = Math.max(dp[j], dp[j - powerOutputs[i]] + powerOutputs[i]);
}
}
return dp[maxPower];
}
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
- 45
- 46
- 47
- 48
C++解法
更新中
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
C解法
更新中
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
JS解法
给定一个数组 powers 表示若干个物品的能力值,以及一个整数 maxPower 表示背包的容量上限。
目标是从 powers 中选择若干物品,使得其能力值总和尽可能接近(但不超过)maxPower,并输出最大总和。
解决方案
使用递归 + 记忆化(动态规划)解决问题。
定义递归函数 findMax(index, remainingPower):
index 表示当前正在考虑的物品索引。
remainingPower 表示背包剩余的容量。
返回在当前状态下能够达到的最大能力值总和。
记忆化数组 memo 用于存储中间结果,避免重复计算:
memo[index][remainingPower] 表示从物品 index 开始、背包剩余容量为 remainingPower 时能够达到的最大能力值。
递归逻辑
终止条件:
如果当前物品索引 index 已超出数组范围,返回 0(表示没有更多物品可选)。
状态转移:
不选择当前物品:结果为 findMax(index + 1, remainingPower)。
选择当前物品(如果其能力值不超过剩余容量):结果为 powers[index] + findMax(index + 1, remainingPower - powers[index])。
最终结果为上述两种选择的最大值。
const readline = require("readline");
const rl = readline.createInterface({
input: process.stdin,
output: process.stdout,
});
const lines = [];
rl.on("line", (line) => {
lines.push(line);
if (lines.length === 3) {
const n = parseInt(lines[0]);
const powers = lines[1].split(" ").map(Number);
const maxPower = parseInt(lines[2]);
console.log(getOptimalPower(n, powers, maxPower));
lines.length = 0;
}
});
function getOptimalPower(n, powers, maxPower) {
const memo = Array(n)
.fill(null)
.map(() => Array(maxPower + 1).fill(-1));
return findMax(0, maxPower, powers, memo);
}
function findMax(index, remainingPower, powers, memo) {
if (index === powers.length) return 0;
if (memo[index][remainingPower] !== -1) return memo[index][remainingPower];
let withoutCurrent = findMax(index + 1, remainingPower, powers, memo);
let withCurrent = 0;
if (powers[index] <= remainingPower) {
withCurrent =
powers[index] +
findMax(index + 1, remainingPower - powers[index], powers, memo);
}
memo[index][remainingPower] = Math.max(withoutCurrent, withCurrent);
return memo[index][remainingPower];
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
- 45
- 46
- 47
- 48
- 49
- 50
- 51
- 52
- 53
- 54
- 55
- 56
- 57
- 58
- 59
- 60
- 61
- 62
- 63
- 64
- 65
- 66
- 67
- 68
- 69
- 70
- 71
- 72
- 73
- 74
- 75
注意:
如果发现代码有用例覆盖不到的情况,欢迎反馈!会在第一时间修正,更新。
解题不易,如对您有帮助,欢迎点赞/收藏
评论记录:
回复评论: