3752:走迷宫

总时间限制: 
1000ms

内存限制: 
65536kB
描述
一个迷宫由R行C列格子组成,有的格子里有障碍物,不能走;有的格子是空地,可以走。
给定一个迷宫,求从左上角走到右下角最少需要走多少步(数据保证一定能走到)。只能在水平方向或垂直方向走,不能斜着走。
输入
第一行是两个整数,R和C,代表迷宫的长和宽。( 1<= R,C <= 40)
接下来是R行,每行C个字符,代表整个迷宫。
空地格子用'.'表示,有障碍物的格子用'#'表示。
迷宫左上角和右下角都是'.'。
输出
输出从左上角走到右下角至少要经过多少步(即至少要经过多少个空地格子)。计算步数要包括起点和终点。
样例输入
5 5
..###
#....
#.#.#
#.#.#
#.#..
样例输出
9

【分析】 本题用递归实现起来不方便,因此用一个结构模拟走迷宫的过程:
1 struct node {
2 int x,y; //节点坐标
3 int step; //到达该节点需要的步数
4 node(int x,int y,int step) : x(x),y(y),step(step) {};
5 };

此处,node表示迷宫中的一个“节点”。

用栈实现dfs走迷宫的思想:(1) 将起始节点入栈;

          (2) 将起始节点出栈,作为“当前所在的节点”

          (3) 将“当前所在的节点”所能到达的所有节点入栈(这便是下一步所有的路);

          (4) 栈顶节点出栈,作为下一步的“当前所在的节点”

          (5) 重复上述过程,直到走到终点。

由于栈后进先出的特点,因此每一个出栈作为“当前所在的节点”的节点一定是上一步刚刚入栈的节点(即上一步的“当前所在的节点”所能直接到达的某个下一
步的节点),因此栈实现的是dfs(而非bfs)。

【代码】
 1 #include <iostream>
2 #include <stack>
3 using namespace std;
4
5 const int maxr = 45;
6 const int maxc = 45;
7 char board[maxr][maxc];
8 bool visited[maxr][maxc];
9 int R,C;
10 int mov[4][2] = { {0,1},{0,-1},{1,0},{-1,0} };
11
12 struct node {
13 int x,y; //节点坐标
14 int step; //到达该节点需要的步数
15 node(int x,int y,int step) : x(x),y(y),step(step) {};
16 };
17
18 bool check(int x,int y)
19 {
20 if( x <= R && x >= 1 && y <= C && y >= 1 && board[x][y] == '.'
21 && !visited[x][y] )
22 return true;
23 return false;
24 }
25
26 int dfs(int x,int y)
27 {
28 stack<node> s;
29 node start(x,y,1);
30 s.push(start);
31
32 //栈仅是工具,真正的动态dfs过程从while循环中才开始,
33 //从栈中取出一个节点,相当于确定了目前所在的节点
34 while( !s.empty() ) {
35 node cur = s.top();
36 s.pop();
37
38 //以下for循环把当前能到达的所有点入栈
39 for( int i=0; i<4; i++ ) {
40 int ix = cur.x + mov[i][0];
41 int iy = cur.y + mov[i][1];
42 if( !check(ix,iy) ) continue;
43
44 node nxt(ix,iy,cur.step+1); //创造下一个合法节点
45
46 //判断:如果已经到目标点直接返回答案
47 if( nxt.x == R && nxt.y == C ) {
48 return nxt.step; //(1)
49 }
50 visited[nxt.x][nxt.y] = true; //不加这句,会导致已走过的节点多次入栈,造成死循环
51 //入栈了的节点今后一定都能访问到,因此不用担心漏掉"可走的路"
52 s.push(nxt);
53 }
54 }
55
56 return -1; //若栈中所有节点都已经取出,程序走到了此处(而不是在(1)处返回),
57 //说明终点无法走到
58 }
59
60 int main()
61 {
62 cin >> R >> C;
63 for( int i=1; i<=R; i++ ) {
64 for( int j=1; j<=C; j++ ) {
65 cin >> board[i][j];
66 }
67 getchar(); //除行尾回车
68 }
69 int ans = dfs(1,1);
70
71 cout << ans << endl;
72 return 0;
73 }

本题使用bfs实现也可以,只需要把栈换成队列就行了。

百炼3752:走迷宫--栈实现dfs的更多相关文章

  1. 数据结构之 栈与队列--- 走迷宫(深度搜索dfs)

    走迷宫 Time Limit: 1000MS Memory limit: 65536K 题目描述 一个由n * m 个格子组成的迷宫,起点是(1, 1), 终点是(n, m),每次可以向上下左右四个方 ...

  2. SDUT-2449_数据结构实验之栈与队列十:走迷宫

    数据结构实验之栈与队列十:走迷宫 Time Limit: 1000 ms Memory Limit: 65536 KiB Problem Description 一个由n * m 个格子组成的迷宫,起 ...

  3. sdut 2449走迷宫【最简单的dfs应用】

    走迷宫 Time Limit: 1000ms   Memory limit: 65536K  有疑问?点这里^_ 题目描述 一个由n * m 个格子组成的迷宫,起点是(1, 1), 终点是(n, m) ...

  4. NYOJ306 走迷宫(dfs+二分搜索)

    题目描写叙述 http://acm.nyist.net/JudgeOnline/problem.php?pid=306 Dr.Kong设计的机器人卡多非常爱玩.它经常偷偷跑出实验室,在某个游乐场玩之不 ...

  5. HDU 2102 A计划(BFS/DFS走迷宫)

    A计划 Time Limit: 3000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submis ...

  6. C语言动态走迷宫

    曾经用C语言做过的动态走迷宫程序,先分享代码如下: 代码如下: //头文件 #include<stdio.h> #include<windows.h>//Sleep(500)函 ...

  7. 洛谷P1238 走迷宫

    洛谷1238 走迷宫 题目描述 有一个m*n格的迷宫(表示有m行.n列),其中有可走的也有不可走的,如果用1表示可以走,0表示不可以走,文件读入这m*n个数据和起始点.结束点(起始点和结束点都是用两个 ...

  8. BZOJ 2707: [SDOI2012]走迷宫( tarjan + 高斯消元 )

    数据范围太大不能直接高斯消元, tarjan缩点然后按拓扑逆序对每个强连通分量高斯消元就可以了. E(u) = 1 + Σ E(v) / degree(u) 对拍时发现网上2个程序的INF判断和我不一 ...

  9. BZOJ 2707: [SDOI2012]走迷宫 [高斯消元 scc缩点]

    2707: [SDOI2012]走迷宫 题意:求s走到t期望步数,\(n \le 10^4\),保证\(|SCC| \le 100\) 求scc缩点,每个scc高斯消元,scc之间直接DP 注意每次清 ...

随机推荐

  1. QFNU-ACM 2019.5.23组队赛 2019山东省赛复现

    A.Calandar 题意:一年12个月,一个月30天,5天一周,已知某天的年月日星期数,求所给年月日的星期数是多少 思路:直接进行计算,其实每个月每年都是等长度的就使得计算的时候忽略年月,可以直接进 ...

  2. apache common pool2原理与实战

    完整源码,请帮我点个star哦! 原文地址为https://www.cnblogs.com/haixiang/p/14783955.html,转载请注明出处! 简介 对象池顾名思义就是存放对象的池,与 ...

  3. 『动善时』JMeter基础 — 15、使用JMeter实现上传文件

    目录 1.用于演示的项目说明 2.测试计划内包含的元件 3.HTTP请求界面内容 4.查看结果 5.总结 6.补充:MIME类型简介 (1)MIME说明 (2)常见类型 在上一篇文章[使用JMeter ...

  4. [网络编程之客户端/服务器架构,互联网通信协议,TCP协议]

    [网络编程之客户端/服务器架构,互联网通信协议,TCP协议] 引子 网络编程 客户端/服务器架构 互联网通信协议 互联网的本质就是一系列的网络协议 OSI七层协议 tcp/ip五层模型 客户端/服务器 ...

  5. str.isdigit()可以判断变量是否为数字

    字符串.isdigit()可以判断变量是否为数字 是则输出True 不是则输出False 好像只能字符串

  6. 【Mysql】数据库事务,脏读、幻读、不可重复读

    一.什么是数据库事务 数据库事务( transaction)是访问并可能操作各种数据项的一个数据库操作序列,这些操作要么全部执行,要么全部不执行,是一个不可分割的工作单位.事务由事务开始与事务结束之间 ...

  7. Linux权限问题(2)-unzip引发的权限问题

    背景:依然是上一个朋友,在用php调用unzip命令时,再次出现了权限被拒绝的问题. Notice:此处描述的问题,为使用php命令行执行php文件,因此进程属主为登录的用户,而不是nginx用户. ...

  8. 本文介绍使用windows系统自带的远程桌面mstsc连接Centos 7.x远程桌面的基本方法。

    本文介绍使用windows系统自带的远程桌面mstsc连接Centos 7.x远程桌面的基本方法. 一.前言 我希望用windows远程访问centos图形界面.xmanager连接centos远程桌 ...

  9. MySQL给某个用户给某个库表设置权限

    -- 用root(最高权限的用户)进行以下操作-- 创建数据库:emc_power CREATE DATABASE emc_power DEFAULT CHARACTER SET utf8 COLLA ...

  10. Linux_计划任务理论概述

    一.计划任务概述 1.计划任务概述: 计划任务分为: 一次性任务 周期性任务 在Linux系统的计划任务服务crond 可以满足周期性执行任务的需求. crond进程每分钟会处璇一次计划任务,计划任务 ...