- 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
java解法
- 解题思路
- 这是一个典型的“过河问题”,其中有两种动物(羊和狼)需要通过一只船过河。每次船上可以载一定数量的羊和狼,并且船的容量有限制。目标是通过最少的步骤,将所有的羊和狼从起始岸带到对岸,并且遵循以下规则:
每次船上带的羊和狼的数量总和不能超过船的容量 b。
每个岸上的羊和狼必须遵守狼不多于羊的规则,除非没有羊。
解题步骤:
状态表示:每个状态可以通过 5 个变量表示:
s: 当前岸上剩余的羊的数量。
w: 当前岸上剩余的狼的数量。
b: 船的最大载客数。
opp_s: 已经成功到达对岸的羊的数量。
opp_w: 已经成功到达对岸的狼的数量。
目标:通过递归方式探索所有可能的搬运方案,最少步骤将所有羊和狼都从起始岸搬到对岸。状态变化是递归进行的,每次通过递归检查合法的羊和狼的搬运数量。
合法性检查:
每次船上带的羊和狼的数量必须小于等于船的容量 b。
在当前岸和对岸,不能让狼的数量超过羊的数量,除非羊的数量为零。
防止重复访问相同的状态,避免无意义的计算。
递归策略:使用深度优先搜索(DFS)策略进行递归搜索,记录每种合法搬运方案的步数,最终返回最小的步数
import java.util.*;
public class Main {
public static void main(String[] args) {
Scanner sc = new Scanner(System.in);
int s = sc.nextInt();
int w = sc.nextInt();
int b = sc.nextInt();
System.out.println(minSteps(s, w, b));
}
public static int minSteps(int s, int w, int b) {
List<Integer> res = new ArrayList<>();
search(s, w, b, 0, 0, 0, res);
return res.isEmpty() ? 0 : Collections.min(res);
}
public static void search(int s, int w, int b, int opp_s, int opp_w, int cnt, List<Integer> res) {
if (s == 0 && w == 0) {
res.add(cnt);
return;
}
if (s + w <= b) {
res.add(cnt + 1);
return;
}
for (int i = 0; i <= Math.min(b, s); i++) {
for (int j = 0; j <= Math.min(b, w); j++) {
if (i + j == 0 || i + j > b) continue;
if (s - i <= w - j && s - i > 0) continue;
if (opp_s + i <= opp_w + j && opp_s + i > 0) continue;
search(s - i, w - j, b, opp_s + i, opp_w + j, cnt + 1, res);
}
}
}
}
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
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解法
每次船上带的羊和狼的数量总和不能超过船的容量 boat。
每个岸上的羊和狼必须遵守狼不多于羊的规则,除非没有羊。
解题步骤:
状态表示:
s: 当前岸上的羊的数量。
w: 当前岸上的狼的数量。
os: 已经成功到达对岸的羊的数量。
ow: 已经成功到达对岸的狼的数量。
boat: 船的容量。
trips: 当前已执行的步数。
目标:通过递归方式探索所有可能的搬运方案,最少步骤将所有羊和狼从起始岸搬到对岸。每次递归都检查当前是否已经将所有羊和狼成功搬到对岸,如果是,则更新最小的步数。
合法性检查:确保在每个状态下,岸上的羊数量不会少于狼的数量,除非羊已经没有了。
递归策略:使用深度优先搜索(DFS)来模拟每次可能的羊和狼的搬运组合,确保每次递归前都验证当前状态是否合法。
const readline = require("readline");
const rl = readline.createInterface({ input: process.stdin, output: process.stdout });
rl.on("line", (line) => {
const [sheep, wolf, boat] = line.split(" ").map(Number);
console.log(minTrips(sheep, wolf, boat));
});
function minTrips(sheep, wolf, boat) {
const res = { min: Infinity };
findTrips(sheep, wolf, 0, 0, boat, 0, res);
return res.min === Infinity ? 0 : res.min;
}
function findTrips(s, w, os, ow, boat, trips, res) {
if (s === 0 && w === 0) {
res.min = Math.min(res.min, trips);
return;
}
for (let i = 0; i <= Math.min(boat, s); i++) {
for (let j = 0; j <= Math.min(boat - i, w); j++) {
if (i + j === 0) continue;
const ns = s - i;
const nw = w - j;
const nos = os + i;
const now = ow + j;
if (validState(ns, nw, nos, now)) {
findTrips(ns, nw, nos, now, boat, trips + 1, res);
}
}
}
}
function validState(s, w, os, ow) {
return !(s > 0 && s < w) && !(os > 0 && os < ow);
}
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
注意:
如果发现代码有用例覆盖不到的情况,欢迎反馈!会在第一时间修正,更新。
解题不易,如对您有帮助,欢迎点赞/收藏
评论记录:
回复评论: