前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >专栏 >算法系列之深度优先搜索寻找妖怪和尚过河问题的所有方式

算法系列之深度优先搜索寻找妖怪和尚过河问题的所有方式

作者头像
修己xj
发布于 2025-03-12 05:07:38
发布于 2025-03-12 05:07:38
8600
代码可运行
举报
文章被收录于专栏:修己xj修己xj
运行总次数:0
代码可运行

在算法学习中,深度优先搜索(DFS)是一种常用的图搜索算法,通过递归或栈实现,适合路径搜索、连通性、拓扑排序、回溯、生成、环路检测、强连通分量和可达性等问题。本文将介绍如何利用深度优先搜索解决“妖怪和尚过河问题”的所有方式。

问题描述

有三个妖怪和三个和尚需要过河。他们只有一条小船,且船最多只能载两人。在任何时候,无论是岸边还是船上,如果妖怪的数量多于和尚,和尚就会被吃掉。如何安排过河顺序,才能让所有妖怪和和尚安全过河?列出所有可能的解?

_20250308_232541.png
_20250308_232541.png

问题分析

首先,我们需要将建立状态和动作的数学模型,我们要明确问题的状态表示。我们用五个属性来表示状态,

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
//左岸和尚数量
int leftMonk;
//左岸妖怪数量
int leftMonster;
//右岸和尚数量
int rightMonk;
//右岸妖怪数量
int rightMonster;
//-1代表左岸,1代表右岸
int boatLocation;

结合船移动的方向,一共右10中过河动作可供选择,分别是

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
* 1、一个妖怪从左岸到右岸
* 2、一个和尚从左岸到右岸
* 3、两个妖怪从左岸到右岸
* 4、两个和尚从左岸到右岸
* 5、一个和尚一个妖怪从左岸到右岸
* 6、一个妖怪从右岸到左岸
* 7、一个和尚从右岸到左岸
* 8、两个妖怪从右岸到左岸
* 9、两个和尚从右岸到左岸
* 10、一个和尚一个妖怪从右岸到左岸

我们的目标是从初始状态 (3, 3, 0, 0, -1) 通过一系列合法的移动,达到目标状态 (0, 0, 3, 3, 1)。

Java使用DFS解决问题

深度优先搜索和广度优先搜索一样,都是对图进行搜索的算法,目的也都是从起点开始搜索,直到到达顶点。深度优先搜索会沿着一条路径不断的往下搜索,直到不能够在继续为止,然后在折返,开始搜索下一条候补路径。我们可以将每个状态看作图中的一个节点,合法的移动就是节点之间的边。通过DFS,我们可以列出所有能过河的方式。

算法步骤

  1. 初始化:将初始状态 (3, 3, 0, 0, -1) 作为递归的起始节点,并标记为已访问。
  2. 递归寻找可能的路径:
  • 取出当前的状态。
  • 生成所有可能的下一步状态。
  • 检查这些状态是否合法(即不违反妖怪和和尚的数量限制)。
  • 如果某个状态是目标状态,则将当前节点添加到过河方式的集合中,回溯的时候将当前节点从访问标记set中删除。
  • 否则,继续递归求解,并标记为已访问。
  1. 递归完成后打印所有可能的过河方式

代码实现如下:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制

/**
 * 妖怪与和尚过河问题
 * 描述:
 * 有三个和尚和三个妖怪在河的左岸,他们需要过河到右岸。
 *
 * 只有一条小船,最多可以承载两个人。
 *
 * 在任何时候,无论是左岸还是右岸,如果和尚的数量少于妖怪的数量,和尚就会被妖怪吃掉。
 *
 * 列出所有可能的解
 */
public class DFSCrossRiver {

    /**
     * 状态类(数据模型)
     */
    public static class State{
        //左岸和尚数量
        int leftMonk;
        //左岸妖怪数量
        int leftMonster;
        //右岸和尚数量
        int rightMonk;
        //右岸妖怪数量
        int rightMonster;
        //-1代表左岸,1代表右岸
        int boatLocation;
        //前一状态,用于回溯打印路径
        State preState;

        public State(int leftMonk, int leftMonster, int rightMonk, int rightMonster, int boatLocation, State preState) {
            this.leftMonk = leftMonk;
            this.leftMonster = leftMonster;
            this.rightMonk = rightMonk;
            this.rightMonster = rightMonster;
            this.boatLocation = boatLocation;
            this.preState = preState;
        }

        @Override
        public boolean equals(Object o) {
            if (this == o) return true;
            if (o == null || getClass() != o.getClass()) return false;
            State state = (State) o;
            return leftMonk == state.leftMonk && leftMonster == state.leftMonster && rightMonk == state.rightMonk && rightMonster == state.rightMonster && boatLocation == state.boatLocation ;
        }

        @Override
        public int hashCode() {
            return Objects.hash(leftMonk, leftMonster, rightMonk, rightMonster, boatLocation);
        }
    }

    /**
     * 动作类,记录妖怪与和尚过河的动作
     */
    @Data
    public static class Action{
        int monkNum; //船上和尚数量
        int monsterNum;//船上妖怪数量
        int boatLocation;//船移动后位置

        public Action(int monkNum, int monsterNum, int boatLocation) {
            this.monkNum = monkNum;
            this.boatLocation = boatLocation;
            this.monsterNum = monsterNum;
        }
    }


    /**
     * 列举过河动作的可供选择
     * 共有十种情况
     * 1、一个妖怪从左岸到右岸
     * 2、一个和尚从左岸到右岸
     * 3、两个妖怪从左岸到右岸
     * 4、两个和尚从左岸到右岸
     * 5、一个和尚一个妖怪从左岸到右岸
     * 6、一个妖怪从右岸到左岸
     * 7、一个和尚从右岸到左岸
     * 8、两个妖怪从右岸到左岸
     * 9、两个和尚从右岸到左岸
     * 10、一个和尚一个妖怪从右岸到左岸
     *
     */
    public static List<Action> getActions(){
        List<Action> actions = new ArrayList<>();
        //一个妖怪从左岸到右岸
        actions.add(new Action(0,1,1));
        //两个妖怪从左岸到右岸
        actions.add(new Action(0,2,1));
        //一个和尚从左岸到右岸
        actions.add(new Action(1,0,1));
        //两个和尚从左岸到右岸
        actions.add(new Action(2,0,1));
        //一个和尚一个妖怪从左岸到右岸
        actions.add(new Action(1,1,1));
        //一个妖怪从右岸到左岸
        actions.add(new Action(0,1,-1));
        //两个妖怪从右岸到左岸
        actions.add(new Action(0,2,-1));
        //一个和尚从右岸到左岸
        actions.add(new Action(1,0,-1));
        //两个和尚从右岸到左岸
        actions.add(new Action(2,0,-1));
        //一个和尚一个妖怪从右岸到左岸
        actions.add(new Action(1,1,-1));
        return actions;

    }

    /**
     * 初始状态
     */
    public static State initState(){
        State state = new State(3,3,0,0,-1,null);
        return state;
    }

    /**
     * 生成所有可能的下一个状态
     */
    public static List<State> generateNextStates(State state){
        List<State> nextStates = new ArrayList<>();
        State nextState;
        for (Action action : getActions()) {
            if(state.boatLocation != action.boatLocation){
                nextState = new State(state.leftMonk - action.monkNum*action.boatLocation, state.leftMonster - action.monsterNum*action.boatLocation,
                        state.rightMonk + action.monkNum*action.boatLocation, state.rightMonster + action.monsterNum*action.boatLocation,
                        action.boatLocation, state);
                //有效则添加
                if(checkState(nextState)){
                    nextStates.add(nextState);
                }
            }
        }
        return nextStates;
    }

    /**
     * 检查状态是否有效,(无论是左岸还是右岸,和尚数量大于妖怪数量则有效)
     */
    public static boolean checkState(State state) {
        //任何一岸的和尚数量不能少于妖怪数量,除非和尚数量为0。
        if(state.leftMonk < 0 || state.leftMonster < 0 || state.rightMonk < 0 || state.rightMonster < 0){
            return false;
        }
        //不管是左岸还是右岸,和尚数量大于妖怪数量或者和尚全部在河对岸则有效,船也只能从对岸来回开
        return (state.leftMonk == 0 || state.leftMonk >= state.leftMonster)
                && (state.rightMonk == 0 || state.rightMonk >= state.rightMonster) && state.boatLocation !=state.preState.boatLocation;
    }

    /**
     * 判断是否成功
     */
    public static boolean isSuccess(State state){
        return state.leftMonk == 0 && state.leftMonster == 0;
    }


    /**
     * 深度优先搜索方式解决渡河问题
     * 添加所有可能的方式
     */
    public static void crossRiver(State state,Set<State> visited,List<State> resultStates){
        //每次递归之前都要检查是否成功,如果成功则添加到结果集合中,并返回。
        if(isSuccess(state)){
            resultStates.add(state);
            return ;
        }
        //递归深度优先搜索
        for(State nextState : generateNextStates(state)){
            if(!visited.contains(nextState)){
                //每种方式都要添加到已访问集合中,避免重复搜索
                visited.add(nextState);
                crossRiver(nextState,visited,resultStates);
                visited.remove(nextState);
            }
        }


    }

    /**
     * 递归打印路径
     */

    public static void  printPath(State state){
        if(state.preState == null){
            return;
        }
        printPath(state.preState);
        System.out.println("从"+(state.preState.boatLocation==-1?"左":"右")+"岸到"+(state.boatLocation==-1?"左":"右")+"岸");
        System.out.println("和尚:"+state.leftMonk+" "+state.rightMonk);
        System.out.println("妖怪:"+state.leftMonster+" "+state.rightMonster);
    }


    public static void main(String[] args) {
        State initState = initState();
        List<State> result = new ArrayList<>();
        Set<State> visited = new HashSet<>();
        visited.add(initState);
        crossRiver(initState,visited,result);
        int i =1;
        for(State state : result){
            System.out.println("--------------方式"+i+"---------------------------");
            printPath(state);
            i++;
        }

    }
}

执行结果如下:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
--------------方式1---------------------------
从左岸到右岸
和尚:3 0
妖怪:1 2
从右岸到左岸
和尚:3 0
妖怪:2 1
从左岸到右岸
和尚:3 0
妖怪:0 3
从右岸到左岸
和尚:3 0
妖怪:1 2
从左岸到右岸
和尚:1 2
妖怪:1 2
从右岸到左岸
和尚:2 1
妖怪:2 1
从左岸到右岸
和尚:0 3
妖怪:2 1
从右岸到左岸
和尚:0 3
妖怪:3 0
从左岸到右岸
和尚:0 3
妖怪:1 2
从右岸到左岸
和尚:0 3
妖怪:2 1
从左岸到右岸
和尚:0 3
妖怪:0 3
--------------方式2---------------------------
从左岸到右岸
和尚:3 0
妖怪:1 2
从右岸到左岸
和尚:3 0
妖怪:2 1
从左岸到右岸
和尚:3 0
妖怪:0 3
从右岸到左岸
和尚:3 0
妖怪:1 2
从左岸到右岸
和尚:1 2
妖怪:1 2
从右岸到左岸
和尚:2 1
妖怪:2 1
从左岸到右岸
和尚:0 3
妖怪:2 1
从右岸到左岸
和尚:0 3
妖怪:3 0
从左岸到右岸
和尚:0 3
妖怪:1 2
从右岸到左岸
和尚:1 2
妖怪:1 2
从左岸到右岸
和尚:0 3
妖怪:0 3
--------------方式3---------------------------
从左岸到右岸
和尚:2 1
妖怪:2 1
从右岸到左岸
和尚:3 0
妖怪:2 1
从左岸到右岸
和尚:3 0
妖怪:0 3
从右岸到左岸
和尚:3 0
妖怪:1 2
从左岸到右岸
和尚:1 2
妖怪:1 2
从右岸到左岸
和尚:2 1
妖怪:2 1
从左岸到右岸
和尚:0 3
妖怪:2 1
从右岸到左岸
和尚:0 3
妖怪:3 0
从左岸到右岸
和尚:0 3
妖怪:1 2
从右岸到左岸
和尚:0 3
妖怪:2 1
从左岸到右岸
和尚:0 3
妖怪:0 3
--------------方式4---------------------------
从左岸到右岸
和尚:2 1
妖怪:2 1
从右岸到左岸
和尚:3 0
妖怪:2 1
从左岸到右岸
和尚:3 0
妖怪:0 3
从右岸到左岸
和尚:3 0
妖怪:1 2
从左岸到右岸
和尚:1 2
妖怪:1 2
从右岸到左岸
和尚:2 1
妖怪:2 1
从左岸到右岸
和尚:0 3
妖怪:2 1
从右岸到左岸
和尚:0 3
妖怪:3 0
从左岸到右岸
和尚:0 3
妖怪:1 2
从右岸到左岸
和尚:1 2
妖怪:1 2
从左岸到右岸
和尚:0 3
妖怪:0 3

总结

通过深度优先搜索,我们可以找到所有可能的过河方案。这个问题展示了如何通过状态空间搜索来解决经典的逻辑问题。希望这篇博客对你理解妖怪与和尚过河问题的解法有所帮助!

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2025-03-09,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 修己xj 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
数据结构与算法-深度优先搜索
深度优先搜索(Depth First Search, DFS)可以理解为走迷宫,假设当一个人走迷宫的时候,会遇到岔路口,面对多条路选择时,可以先随便选择一条,走着走着发现如果走不通了,可以退回到上一个岔路口,然后重新选择一条,用同样的方法继续走,直到直到出口为止。这样的策略即为DFS。
用户3470542
2019/06/26
6550
数据结构与算法-深度优先搜索
LeetCode-算法- 广度和深度优先搜索-第20天
给你一个由 '1'(陆地)和 '0'(水)组成的的二维网格,请你计算网格中岛屿的数量。岛屿总是被水包围,并且每座岛屿只能由水平方向和/或竖直方向上相邻的陆地连接形成。此外,你可以假设该网格的四条边均被水包围。
布衣者
2021/09/07
2340
算法系列之深度/广度优先搜索解决水桶分水的最优解及全部解
在算法学习中,广度优先搜索(BFS)适用于解决最短路径问题、状态转换问题等。深度优先搜索(DFS)适合路径搜索等问题。本文将介绍如何利用广度优先搜索解决寻找3 个 3、5、8 升水桶均分 8 升水的最优解及深度优先搜索寻找可以解决此问题的所有解决方案。
修己xj
2025/03/12
950
算法系列之深度/广度优先搜索解决水桶分水的最优解及全部解
算法系列之深度/广度优先搜索解决水桶分水的最优解及全部解
在算法学习中,广度优先搜索(BFS)适用于解决最短路径问题、状态转换问题等。深度优先搜索(DFS)适合路径搜索等问题。本文将介绍如何利用广度优先搜索解决寻找3 个 3、5、8 升水桶均分 8 升水的最优解及深度优先搜索寻找可以解决此问题的所有解决方案。
修己xj
2025/03/12
600
算法系列之深度/广度优先搜索解决水桶分水的最优解及全部解
常用的搜索算法之深度优先搜索
深度优先搜索(DFS)是一种用于遍历或搜索树或图的算法。该算法会尽可能深地搜索树的分支。当节点v的所在边都已被探寻过,搜索将回溯到发现节点v的那条边的起始节点。这个过程一直进行到已发现从源节点可达的所有节点为止。如果还存在未被发现的节点,则选择其中一个作为源节点并重复以上过程,整个进程反复进行直到所有节点都被访问为止。
jack.yang
2025/04/05
1110
java算法刷题02——深度优先搜索与广度优先搜索
给一个01矩阵,1代表是陆地,0代表海洋, 如果两个1相邻,那么这两个1属于同一个岛。我们只考虑上下左右为相邻。 岛屿: 相邻陆地可以组成一个岛屿(相邻:上下左右) 判断岛屿个数。 例如: 输入 [ [1,1,0,0,0], [0,1,0,1,1], [0,0,0,1,1], [0,0,0,0,0], [0,0,1,1,1] ] 对应的输出为3 示例1 输入:
半旧518
2022/10/26
6380
java算法刷题02——深度优先搜索与广度优先搜索
Python算法——深度优先搜索(DFS)
深度优先搜索(Depth-First Search,DFS)是一种遍历或搜索树、图等数据结构的算法。在DFS中,我们从起始节点开始,沿着一条路径尽可能深入,直到达到树的末端或图中的叶子节点,然后回溯到前一节点,继续深入下一路径。这一过程不断重复,直到所有节点都被访问。在本文中,我们将详细讨论DFS的原理,并提供Python代码实现。
Echo_Wish
2023/11/30
1.1K0
算法系列之搜索算法-深度优先搜索DFS
随着每年"金三银四"招聘季的到来,许多求职者开始积极备战面试。在众多面试环节中,机试往往是不可或缺的一环,而算法能力更是机试考核的重点。为此,我们特别推出算法系列文章,帮助大家系统复习算法知识。今天,我们将首先探讨搜索算法中的重要内容——深度度优先搜索(DFS)。
修己xj
2025/03/12
1200
算法系列之搜索算法-深度优先搜索DFS
算法细节系列(16):深度优先搜索
版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/u014688145/article/details/71630857
用户1147447
2019/05/26
4210
Go 每日一库之 bitset
我们都知道计算机是基于二进制的,位运算是计算机的基础运算。位运算的优势很明显,CPU 指令原生支持、速度快。基于位运算的位集合在有限的场景中替换集合数据结构可以收到意想不到的效果。bitset库实现了位集合及相关操作,不妨拿来即用。
用户7731323
2022/11/16
4460
深度优先搜索算法在图论领域的应用与实现
【玩转 GPU】AI绘画、AI文本、AI翻译、GPU点亮AI想象空间-腾讯云开发者社区-腾讯云 (tencent.com)
疯狂的KK
2023/07/10
3440
深度优先搜索算法在图论领域的应用与实现
三分钟讲明白DFS(深度优先搜索)
稍微了解一点的人都知道,当我们需要从一个树结构中寻找到一些符合条件的元素时,我们都知道通过广度优先搜索或者深度优先搜索来有效地解决问题。那么具体是怎样一种手段去搜索呢?广度优先搜索(BFS)我们之前已经聊过了,现在我们就来谈谈深度优先搜索(DFS)。
写代码的阿宗
2020/08/24
6860
【数据结构与算法】图遍历算法 ( 深度优先搜索代码示例 )
文章目录 一、深度优先搜索算法 二、完整代码示例 完整代码示例 执行结果 一、深度优先搜索算法 ---- 深度优先搜索算法步骤 : 将 深度优先搜索 算法步骤 转为代码 ; ① 访问初始结点 : 访问 初始结点 v , 并将该 初始结点 v 标记为 " 已访问 " ; 设置一个 访问标记 数组 , 数组元素个数与 顶点个数相同 ; /** * 判定顶点是否被访问 */ private boolean[] isVisted; ② 查找邻接节点 : 查找 初始结点 v 的 第一个 邻接节点 w
韩曙亮
2023/03/30
3790
【数据结构与算法】图遍历算法 ( 深度优先搜索代码示例 )
图的遍历之深度优先搜索(DFS)
深度优先搜索(depth-first search)是对先序遍历(preorder traversal)的推广。”深度优先搜索“,顾名思义就是尽可能深的搜索一个图。想象你是身处一个迷宫的入口,迷宫中的
llhthinker
2018/01/24
1.9K0
从简单二叉树问题重新来看深度优先搜索
对于一般的二叉树问题,我们总能想到的是深度优先搜索这个算法,继续想下去就是递归,但是其实对于深度优先搜索,有很多不一样的思考方向和实现细节,在这基础上,我们可以推导、总结出一些其他的高级算法,例如分治、动态规划等等,把这些算法联系在一起,更有助于我们理解一些核心的、本质的问题。
五分钟学算法
2019/06/20
6390
【人工智障入门实战1】使用深度优先搜索实现 Amazing-Brick 小游戏的自动控制
•搜索:精准预测下一步操作后,黑色方块将到达什么位置;并再次精准预测在这个位置进行操作后,黑色方块将到达什么位置...直到触发终止条件,即找到最终得分的路径;•深度优先:假设黑色方块有两个动作可以选择:A与B,那么黑色方块做出“选择A后应该到达的位置”的预测后,继续接着这条路径预测,而非去预测在初始状态下“选择B后应该到达的位置”。具体原理如下图。
Piper蛋窝
2020/11/19
5990
【人工智障入门实战1】使用深度优先搜索实现 Amazing-Brick 小游戏的自动控制
深度优先算法
在这段时间中,我一直在看算法问题,在进行刷题;前面还提到了贪心算法,解释了什么是贪心算法,并使用其算法解决了一些算法问题。
半月无霜
2025/01/26
840
Python算法解析:深度优先搜索的魅力与实现策略!
深度优先搜索(DFS)是一种用于图或树的遍历算法,它沿着路径直到无法继续前进,然后回退到前一个节点,继续探索其他路径。
测试开发囤货
2023/08/08
2770
Python算法解析:深度优先搜索的魅力与实现策略!
DFS深度优先搜索解决迷宫问题
的网格迷宫G。G的每个格子要么是道路,要么是障碍物(道路用1表示,障碍物用2表示)。
别团等shy哥发育
2023/03/14
8860
DFS深度优先搜索解决迷宫问题
深度优先搜索
以上递归实现斐波那契实际上就是按照深度优先的方式进行搜索。也就是 “一条路走到黑” 。注意:这里的搜索指的是一种穷举方式,把可行的方案都列举出来,不断尝试,直到找到问题的解。
可爱见见
2020/02/25
9300
深度优先搜索
推荐阅读
相关推荐
数据结构与算法-深度优先搜索
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验