小老鼠走进了格子迷宫,如何能绕过猫并以最短的路线吃到奶酪呢?

注意只能上下左右移动,不能斜着移动。

在解决迷宫问题上,深度优先算法的思路是沿着一条路一直走,遇到障碍或走出边界再返回尝试别的路径。

首先用一个二维数组来把迷宫“数字化”。

  1. int[][] maze = new int[5][4];

迷宫中每个格子的横纵坐标对应数组的一维和二维索引,例如最左上角的格子是maze[0][0],数组的值表示该格子是否可以通过,0表示可以通过,1表示该格子有猫。

初始化迷宫,标记猫的位置:

  1. this.maze[2][0] = 1;
  2. this.maze[1][2] = 1;
  3. this.maze[2][2] = 1;
  4. this.maze[3][2] = 1;

起点位置坐标是x=0,y=0,如果向右移动就是x=x+1,y=y,向下移动是x=x,y=y+1。我们预先规定每到一个格子都按照右、下、左、上的顺序尝试下一个格子是否能走,如果右边的格子没有猫且未出边界,就移动到下一个格子,继续按照右、下、左、上的顺序尝试;如果右边的格子不能走则尝试下面的格子。

下面这个二维数组用来遍历尝试四个方向的格子:

  1. int[][] next = new int[][] {
  2. {1, 0},
  3. {0, 1},
  4. {-1, 0},
  5. {0, -1}
  6. };

为了不走回头路,我们还需要另外一个二维数组标记哪些格子是已走过的,如果已走过则不能回头。

  1. int[][] mark = new int[5][4];

用一个栈记录路径

  1. LinkedList<Integer> map = new LinkedList<>();

走格子的思路是:

  1. for(遍历四个方向的格子) {
  2. if(格子超出边界 或 格子有猫 或 格子已经走过) {
  3. continue;
  4. } else {
  5. 移动到格子
  6. 记录当前格子已走过
  7. 记录当前路径
  8. for(以新格子为中心遍历四个方向的格子) {
  9. ......
  10. }
  11. }
  12. }

但是我们并不知道要走多少步才能到达目标,也就不知道循环要嵌套多少层,但是可以看出每次新的遍历循环开启后,执行的代码和上一层循环是一样的,所以这里用递归解决。来看完整的代码:

  1. import java.util.LinkedList;
  2. public class DfsRatMaze {
  3. int min = Integer.MAX_VALUE;
  4. int endX = 3;  //目标点横坐标
  5. int endY = 3;  //目标点纵坐标
  6. int width = 5;  //迷宫宽度
  7. int height = 4;  //迷宫高度
  8. int[][] maze = new int[5][4];
  9. int[][] mark = new int[5][4];
  10. LinkedList<Integer> map = new LinkedList<>();
  11. public void dfs(int startX, int startY, int step) {
  12. int[][] next = new int[][] { //按右->下->左->上的顺序尝试
  13. {1, 0},
  14. {0, 1},
  15. {-1, 0},
  16. {0, -1}
  17. };
  18. int nextX, nextY;
  19. int posible;
  20. if(startX == endX && startY == endY) {
  21. if(step < min)
  22. min = step;
  23. for(int i = map.size() - 1; i >= 0; i -= 2){
  24. nextX = map.get(i);
  25. nextY = map.get(i - 1);
  26. System.out.print("[" + nextX + "," + nextY + "]");
  27. if(i != 1)
  28. System.out.print("->");
  29. }
  30. System.out.println();
  31. return;
  32. }
  33. for(posible = 0; posible < next.length; posible++) { //按右->下->左->上的顺序尝试
  34. nextX = startX + next[posible][0];
  35. nextY = startY + next[posible][1];
  36. if(nextX < 0 || nextX >= width || nextY < 0 || nextY >= height) {  //超出边界
  37. continue;
  38. }
  39. if(maze[nextX][nextY] == 0 && mark[nextX][nextY] == 0) {  //非障碍且未标记走过
  40. map.push(nextX);
  41. map.push(nextY);
  42. mark[nextX][nextY] = 1;
  43. dfs(nextX, nextY, step + 1);  //递归调用, 移动到下一格
  44. mark[nextX][nextY] = 0;
  45. map.pop();
  46. map.pop();
  47. }
  48. }
  49. }
  50. /*
  51. * 初始化迷宫
  52. */
  53. public void initMaze() {
  54. this.maze = new int[width][height];
  55. this.mark = new int[width][height];
  56. this.maze[2][0] = 1;
  57. this.maze[1][2] = 1;
  58. this.maze[2][2] = 1;
  59. this.maze[3][2] = 1;
  60. this.mark[0][0] = 1;
  61. //打印迷宫 _表示可通行 *表示障碍 !表示目标
  62. for(int y = 0; y < height; y++) {
  63. for(int x = 0; x < width; x++) {
  64. if(x == endX && y == endY) {
  65. System.out.print("!  ");
  66. }  else if(this.maze[x][y] == 1) {
  67. System.out.print("*  ");
  68. } else {
  69. System.out.print("_  ");
  70. }
  71. }
  72. System.out.println();
  73. }
  74. System.out.println();
  75. }
  76. public static void main(String[] args) {
  77. int startX = 0;
  78. int startY = 0;
  79. DfsRatMaze d = new DfsRatMaze();
  80. d.initMaze();
  81. d.dfs(startX, startY, 0);
  82. if(d.min < Integer.MAX_VALUE)
  83. System.out.println("最少需要" + d.min + "步");
  84. else
  85. System.out.println("目标地点无法到达");
  86. }
  87. }

运行后输出:

  1. [1,0]->[1,1]->[2,1]->[3,1]->[4,1]->[4,2]->[4,3]->[3,3]
  2. [1,0]->[1,1]->[2,1]->[3,1]->[3,0]->[4,0]->[4,1]->[4,2]->[4,3]->[3,3]
  3. [1,0]->[1,1]->[0,1]->[0,2]->[0,3]->[1,3]->[2,3]->[3,3]
  4. [0,1]->[1,1]->[2,1]->[3,1]->[4,1]->[4,2]->[4,3]->[3,3]
  5. [0,1]->[1,1]->[2,1]->[3,1]->[3,0]->[4,0]->[4,1]->[4,2]->[4,3]->[3,3]
  6. [0,1]->[0,2]->[0,3]->[1,3]->[2,3]->[3,3]
  7. 最少需要6步

可以看到,程序计算出了所有路线,并找到了最短的路线。而整个代码还不到100行,真是神奇的算法。

Java与算法之(5) - 老鼠走迷宫(深度优先算法)的更多相关文章

  1. 老鼠走迷宫I

    转自:http://blog.csdn.net/holymaple/article/details/8582517 说明:老鼠走迷宫是递回求解的基本提醒,我们在二维阵列中使用2来表示迷宫墙壁,使用1来 ...

  2. 老鼠走迷宫II

    转自:http://blog.csdn.net/holymaple/article/details/8636234 由于迷宫的设计,老鼠走迷宫的入口至出口路径可能不止一条,如何求出所有的路径呢? 解法 ...

  3. noj->电子老鼠走迷宫

    00 问题 描述: 有一只电子老鼠被困在如下图所示的迷宫中.这是一个12*12单元的正方形迷宫,黑色部分表示建筑物,白色部分是路.电子老鼠可以在路上向上.下.左.右行走,每一步走一个格子.现给定一个起 ...

  4. noj电子老鼠走迷宫(深搜dfs)超时错误

    1042.电子老鼠闯迷宫 时限:1000ms 内存限制:10000K  总时限:3000ms 描述 有一只电子老鼠被困在如下图所示的迷宫中.这是一个12*12单元的正方形迷宫,黑色部分表示建筑物,白色 ...

  5. 老鼠走迷宫(2)输出所有路径(C语言)

    需求 有一个迷宫,在迷宫的某个出口放着一块奶酪.将一只老鼠由某个入口处放进去,它必须穿过迷宫,找到奶酪.请找出它的行走路径. STEP 1 题目转化 我们用一个二维数组来表示迷宫,用2表示迷宫的墙壁, ...

  6. 老鼠走迷宫(1)输出唯一路径(C语言)

    需求 有一个迷宫,在迷宫的某个出口放着一块奶酪.将一只老鼠由某个入口处放进去,它必须穿过迷宫,找到奶酪.请找出它的行走路径. STEP 1 题目转化 我们用一个二维数组来表示迷宫,用2表示迷宫的墙壁, ...

  7. Java与算法之(12) - 老鼠再闯迷宫(广度优先算法)

    贪吃的小老鼠又回来了,这次有什么新的办法吃到奶酪呢? 规则不变,只能上下左右在格子内移动. 因为上次的深度优先算法让老鼠走了不少冤枉路,这次老鼠带来了帮手探路鼠.探路鼠的使用规则如下: 小老鼠按右.下 ...

  8. Java基于OpenCV实现走迷宫(图片+路线展示)

    Java基于OpenCV实现走迷宫(图片+路线展示) 由于疫情,待在家中,太过无聊.同学发了我张迷宫图片,让我走迷宫来缓解暴躁,于是乎就码了一个程序出来.特此记录. 原图: 这张图,由于不是非常清晰, ...

  9. golang 实现广度优先算法(走迷宫)

    maze.go package main import ( "fmt" "os" ) /** * 广度优先算法 */ /** * 从文件中读取数据 */ fun ...

随机推荐

  1. 判断json数据是否包含key

    1.("key" in jsonObj) 如果有返回true 没有返回false 2.jsonObj.hasOwnProperty("key") 如果有返回tr ...

  2. Laravel学习笔记(一)

         根据国外的调查,Laravel是最流行的框架,最近公司需要PHP的开发人员,但是一直招不到人,只好亲自上阵研究一下.由于以前对PHP只是大致了解,这次学习开始的时候也挺挠头的,到今天稍微入了 ...

  3. JaveScript运算符(JS知识点归纳三)

    JaveScript中有许多的运算符,在这里就只说明一些需要注意的. 01 一元运算符 一元:指的是参与运算的操作数只有一个 最经常使用的是++   -- 计算规则: ++/-- 前置于操作数的时候 ...

  4. iOS UIWebView 加载进度条的使用-WKWebView的使用,更新2017.6.26

    1.由于项目中加载网络插件,直接使用了webview加载.使用了三方NJKWebViewProgress进度条的使用,近期在测试时发现,网络缓慢时出现白屏,有卡顿现象. 于是采用了WKWebView进 ...

  5. Docker Stack 集群部属服务

    Docker越来越成熟,功能也越来越强大.使用Dokcer Stack做服务集群也是非常的方便,docker 自己就提供了负载功能,感觉很方便,就想给大家分享一下,做一个简单的教程. 环境 我是用了两 ...

  6. ubuntu12.04destdrop删除不必要的软件

    sudo apt-get -y --auto-remove purge unity unity-2d*  sudo apt-get -y purge empathy  sudo apt-get -y ...

  7. [摘抄]VC6.0移植到VS2008(vs2005)后的错误总结(未全部验证)

    ============================================================================================= 201405 ...

  8. 高级开发层面,针对Hibernate方面面试题的总结(对其它ORM也适用)

    虽然目前mytabis用得比较多,但Hibernate相对比较容易上手,而且也有不少公司在用,所以本文就用这个举例,事实上,本文给出的面试建议也适用于各种ORM.本文摘自java web轻量级开发面试 ...

  9. 基于Dubbo的http自动测试工具分享

    公司是采用微服务来做模块化的,各个模块之间采用dubbo通信.好处就不用提了,省略了之前模块间复杂的http访问.不过也遇到一些问题: PS: Github的代码示例还在整理中... 测试需要配合写消 ...

  10. SQL2005 到 SQL2008R2 发布订阅----发布'xxxxx'的初始快照尚不可用。

    步骤略! SQL2005 到 SQL2008R2 发布订阅----发布'xxxxx'的初始快照尚不可用. 发布库快照已经创建完成为什么到订阅就快照不可用呢! 订阅通过日志读取代理解析! 查了下代理安全 ...