当前位置: 首页 > news >正文

[蓝桥杯]迷宫与陷阱

迷宫与陷阱

题目描述

小明在玩一款迷宫游戏,在游戏中他要控制自己的角色离开一间由 N×NN×N 个格子组成的 2D 迷宫。

小明的起始位置在左上角,他需要到达右下角的格子才能离开迷宫。

每一步,他可以移动到上下左右相邻的格子中(前提是目标格子可以经过)。

迷宫中有些格子小明可以经过,我们用 '.' 表示。

有些格子是墙壁,小明不能经过,我们用 '#' 表示。

此外,有些格子上有陷阱,我们用 'X' 表示。除非小明处于无敌状态,否则不能经过。

有些格子上有无敌道具,我们用 '%' 表示。

当小明第一次到达该格子时,自动获得无敌状态,无敌状态会持续 KK 步。

之后如果再次到达该格子不会获得无敌状态了。

处于无敌状态时,可以经过有陷阱的格子,但是不会拆除/毁坏陷阱,即陷阱仍会阻止没有无敌状态的角色经过。

给定迷宫,请你计算小明最少经过几步可以离开迷宫?

输入描述

输入描述

第一行包含两个整数 N,K (1≤N≤1000,1≤K≤10)N,K (1≤N≤1000,1≤K≤10)。

以下 NN 行包含一个 N×NN×N 的矩阵。

矩阵保证左上角和右下角是 '.'。

输出描述

一个整数表示答案。如果小明不能离开迷宫,输出 -1。

输入输出样例

示例

输入

5 3
...XX
##%#.
...#.
.###.
.....

输出

10

运行限制

  • 最大运行时间:3s
  • 最大运行内存: 256M

总通过次数: 1649  |  总提交次数: 2154  |  通过率: 76.6%

难度: 困难   标签: 2018, 国赛, BFS

迷宫与陷阱问题:带状态BFS算法详解

算法思路

本问题需要在经典BFS基础上增加​​无敌状态​​的维度,核心思路是:

  1. ​状态表示​​:每个位置(x,y)叠加当前剩余无敌步数s,形成三元组(x, y, s)。不同无敌步数的同一位置视为不同状态。
  2. ​无敌状态规则​​:
    • 陷阱X:仅当无敌步数s>0时可通行,通过后步数减1
    • 道具%:首次到达时重置无敌步数为K(覆盖原有状态),后续到达仅继承剩余步数
  3. ​BFS扩展​​:从队列取出状态后,向四个方向扩展新状态:
    • 计算基础无敌步数ns_base = (s>0 ? s-1 : 0)
    • 根据格子类型调整最终无敌步数ns
  4. ​终止条件​​:首次到达终点(N-1, N-1)时返回步数(BFS保证最小步数)
代码实现
#include <iostream>
#include <cstring>
using namespace std;const int MAXN = 1005;
const int MAXK = 11; // K∈[1,10] 状态数0~K共11种
const int MAXS = MAXN * MAXN * MAXK; // 最大状态数int n, K;
char grid[MAXN][MAXN];
bool got[MAXN][MAXN];   // 道具是否已获取
int dist[MAXN][MAXN][MAXK]; // dist[x][y][s] = 最小步数
int qx[MAXS], qy[MAXS], qs[MAXS]; // 数组模拟队列
int dx[4] = {1, 0, -1, 0};
int dy[4] = {0, 1, 0, -1};int bfs() {memset(dist, -1, sizeof(dist));memset(got, false, sizeof(got));int front = 0, rear = 0;// 初始化起点dist[0][0][0] = 0;qx[rear] = 0;qy[rear] = 0;qs[rear] = 0;rear++;while (front < rear) {int x = qx[front];int y = qy[front];int s = qs[front];front++;int step = dist[x][y][s];// 到达终点if (x == n-1 && y == n-1) return step;// 向四个方向扩展for (int i = 0; i < 4; i++) {int nx = x + dx[i];int ny = y + dy[i];// 边界检查if (nx < 0 || nx >= n || ny < 0 || ny >= n) continue;// 遇到墙壁if (grid[nx][ny] == '#') continue;// 遇到陷阱且无无敌状态if (grid[nx][ny] == 'X' && s == 0) continue;// 计算基础无敌步数(移动消耗1步)int ns_base = (s > 0) ? s - 1 : 0;int ns = ns_base; // 默认值// 道具格子处理if (grid[nx][ny] == '%') {if (!got[nx][ny]) {got[nx][ny] = true; // 标记道具已获取ns = K;             // 重置无敌状态}}// 状态未访问过则入队if (dist[nx][ny][ns] == -1) {dist[nx][ny][ns] = step + 1;qx[rear] = nx;qy[rear] = ny;qs[rear] = ns;rear++;}}}return -1; // 无法到达终点
}int main() {cin >> n >> K;for (int i = 0; i < n; i++)cin >> grid[i];cout << bfs() << endl;return 0;
}
代码解析
  1. ​状态存储​​:

    • dist[x][y][s]:记录到达(x,y)且剩余无敌步数s的最小步数
    • got[x][y]:标记道具格子是否已被获取
    • 数组qx, qy, qs模拟队列,避免STL开销
  2. ​核心逻辑​​:

    • ​陷阱处理​​(L38-40):仅当当前状态s>0时允许通过
    • ​道具重置​​(L47-51):首次到达时重置无敌步数为K
    • ​状态更新​​(L54-60):仅当新状态未访问过时入队
  3. ​终止条件​​(L28-29):首次到达终点立即返回(BFS性质保证最小步数)

实例验证
输入:
5 3
...XX
##%#.
...#.
.###.
.....输出:10

​执行过程解析​​:

  1. 起点(0,0,0) → 向右到(0,1,0)(步数1)
  2. 继续向右到(0,2,0) → 向下到道具格(1,2,3)(重置无敌,步数3)
  3. 利用无敌状态穿越陷阱:
    • (1,2,3) → (2,2,2) → (3,2,1) → (4,2,0)
  4. 向下移动:
    • (4,2,0) → (4,3,0) → (4,4,0)(步数10)

路径可视化:
https://example.com/maze_path.png
绿色路径为最优解,红色数字表示步数

注意事项
  1. ​状态去重​​:

    • 每个状态(x,y,s)只扩展一次(dist数组保证)
    • 同一位置不同无敌步数视为独立状态
  2. ​道具全局性​​:

    • got数组标记道具获取状态(全局唯一)
    • 重置无敌状态会覆盖原有步数(非叠加)
  3. ​边界情况​​:

    • 起点和终点保证为'.'
    • K=1时需测试无敌状态传递
    • 终点可能以不同无敌步数到达(取首次到达)
测试点设计
测试类型输入样例预期输出验证要点
基础样例题目给定示例10标准流程
无解情况全图被陷阱包围-1终止条件
道具重置多个道具格最短路径全局got标记
无敌传递K=1时连续过陷阱正确步数状态递减逻辑
边界规模N=1000, K=10的大地图不超3s性能验证
优化建议
  1. ​内存压缩​​:

    • 使用short dist[MAXN][MAXN][MAXK](步数<2000)
    • 用位运算压缩got数组(每布尔值占1bit)
  2. ​搜索优化​​:

    • 双向BFS(起点终点同时搜索)
    • 优先扩展靠近终点的状态(A*启发式)
  3. ​常数优化​​:

    • 循环展开(手动展开方向循环)
    • 使用局部变量缓存数组访问

​复杂度分析​​:
时间:O(N²K) ≈ 1000²×11 = 1100万次操作
空间:O(N²K) ≈ 44MB(dist) + 132MB(队列)

http://www.lqws.cn/news/190279.html

相关文章:

  • 排序算法总结(C++)
  • ansible和saltstack安装和简单操作
  • Python训练营打卡DAY46
  • EtherNet/IP转DeviceNet协议网关详解
  • 悲观锁和乐观锁
  • 命令行以TLS/SSL显式加密方式访问FTP服务器
  • 【Linux】ps 命令详解及使用示例:查看当前运行进程状态
  • Linux配置yum 时间同步服务 关闭防火墙 关闭ESlinux
  • 《C语言·源初法典》---C语言基础(上)
  • python fbx sdk
  • C/C++ 中附加包含目录、附加库目录与附加依赖项详解
  • placeholder不显示and模板字符串无效
  • leetcode sql50题
  • clickhouse 学习总结
  • Charles 全流程指南:安装、设置、抓包与注意事项
  • Redis知识
  • 缓解骨质疏松 —— 补钙和补维 D
  • LangChain【8】之工具包深度解析:从基础使用到高级实践
  • C++.OpenGL (6/64)坐标系统(Coordinate Systems)
  • C++单例模式教学指南
  • 2003-2024年高铁列车信息数据
  • PP-OCRv5_server_det.yml参数解释
  • 【PDF PicKiller】PDF批量删除固定位置图片工具,默认解密,可去一般图、背景图、水印图!
  • 图片切割工具:智能分割长图并控制文件大小
  • 谷歌云代理商 | 游戏行业专属方案:谷歌云实时多人游戏服务器架构
  • 使用 Docker Compose 从零部署 TeamCity + PostgreSQL(详细新手教程)
  • 35.成功解决编写关于“江协科技”编写技巧第二期标志位积累的问题
  • vue3学习(toRefs和toRef,computed计算属性 ,v-model指令,箭头函数)
  • 【leetcode】3. 无重复字符的最长子串
  • 跟我学c++中级篇——理解类型推导和C++不同版本的支持