逃离妖洞

描述

唐僧不小心又掉入妖怪的迷宫了。这个迷宫有n行m列,共n*m个方格。有的方格是空的,唐僧可以站在上面,有些是有障碍物的不能站。每次唐僧可以移动到相邻的8个空方格上。但是有些情况不能移动,即从两个方格的缝隙中移动,如图:

为了逃出妖怪的洞穴,唐僧需要触发悟空在某个方格留下的宝物,宝物都是留在空的方格中的。如果唐僧站在某个留有宝物的方格中,那么这个宝物就会被触发。为了逃离妖洞,唐僧务必按顺序依次触发K个宝物,当最后一个宝物被触发后,唐僧将会逃离妖洞。

现在唐僧已经知道这个迷宫中的障碍方格和有宝物的方格情况,开始唐僧站在一个给定的方格。问他是否可以逃离妖洞?

输入
一共有T(T<=20)组测试数据。



每组含有行数n,列数m。(2<=n,m<=100)以及放有宝物的K(1<= K <=10)个方格。



下面n行,每行包含m个字符,用'.'表示空的方格,用'#'表示有障碍物的方格。



下面一行是起点位置(x,y),保证是空的方格。

下面的K行表示藏有宝物的空方格的位置,藏有宝物的方格是空的,没有障碍物。两个宝物不会放在同一个空方格中。需要按所给宝物顺序依次触发宝物。
输出
输出为了逃离妖洞,最少需要多少次移动。如果不能逃离,输出-1。
样例输入
3
3 3 2
...
...
...
1 1
1 3
2 2
3 3 1
...
.#.
...
1 1
3 3
2 3 1
..#
.#.
1 1
2 3
样例输出
3
3
-1

值得注意的地方

1.  宝物必须依次触发,也就是说不能提前通过未触发的宝物。

2. 起点可能位于某个宝物上面,如果不是位于第1个宝物的坐标,那么唐僧不可能逃离的。

AC代码:

#include<cstdio>
#include<cstring>
#include<queue>
using namespace std;

const int maxn = 100 + 5;
int d[maxn][maxn];
char G[maxn][maxn];
int n, m, k;         

const int dx[] = {1,-1,0,0, 1,-1,-1,1};
const int dy[] = {0,0,1,-1, -1,-1,1,1};

struct node{
	int x, y;
	node(){
	}
	node(int x, int y):x(x), y(y){
	}
}h[15];

int bfs(int x, int y, char g){
	memset(d, -1, sizeof(d));
	queue<node>q;
	q.push(node(x, y));
	d[x][y] = 0;
	while(!q.empty()){
		node p = q.front();
		q.pop();
		x = p.x, y = p.y;
		if(G[x][y] == g) return d[x][y];
		for(int i = 0; i < 8; ++i){
			int px = x + dx[i], py = y + dy[i];
			if(px < 0 || px >= n || py < 0 || py >= m || G[px][py] == '#' || G[px][py] > g || d[px][py] != -1) continue;
			if(i >= 4 && G[x + dx[i]][y] == '#' && G[x][y + dy[i]] == '#') continue;
			q.push(node(px, py));
			d[px][py] = d[x][y] + 1;
		}
	}
	return -1;
}

int solve(int x, int y){
	if(G[x][y] == '#') return -1;
	if(G[x][y] != '.' && G[x][y] > '0') return -1;
	int ans = 0;
	for(int i = 0; i < k; ++i){
		int step = bfs(x, y, '0' + i);
		if(step == -1) return -1;
		ans += step;
		x = h[i].x, y = h[i].y;
	}
	return ans;
}

int main(){
	int T;
	scanf("%d", &T);
	while(T--){
		scanf("%d%d%d", &n, &m, &k);
		for(int i = 0; i < n; ++i){
			scanf("%s", G[i]);
		}
		int x, y;
		scanf("%d%d", &x, &y);
		int x1, y1; //宝物的坐标
		for(int i = 0; i < k; ++i){
			scanf("%d%d", &x1, &y1);
			h[i] = node(x1 - 1, y1 - 1);
			G[x1 - 1][y1 - 1] = i + '0';
		}
		printf("%d\n",solve(x - 1, y - 1));
	}
	return 0;
}

如有不当之处欢迎指出!

nyoj1246 逃离妖洞 BFS的更多相关文章

  1. HDU 1728:逃离迷宫(BFS)

    http://acm.hdu.edu.cn/showproblem.php?pid=1728 逃离迷宫 Problem Description   给定一个m × n (m行, n列)的迷宫,迷宫中有 ...

  2. hdu1072 逃离迷宫系列 bfs

    题目链接:http://icpc.njust.edu.cn/Problem/Hdu/1072/ 题意:逃离迷宫,路中可能有炸弹,总时间是6个单位,在有炸弹的位置,如果到达的时刻时间大于0,则恢复到6时 ...

  3. HDU 1728 逃离迷宫(BFS)

    Problem Description 给定一个m × n (m行, n列)的迷宫,迷宫中有两个位置,gloria想从迷宫的一个位置走到另外一个位置,当然迷宫中有些地方是空地,gloria可以穿越,有 ...

  4. [HIHO1328]逃离迷宫(bfs,位压)

    题目链接:http://hihocoder.com/problemset/problem/1328 这个题bfs到时候不止要存当前的坐标,还要存当前有哪几把钥匙.因为5把钥匙,所以可以直接用位来存,这 ...

  5. HDU 1728 逃离迷宫【BFS】

    题意:给出一个起点,一个终点,规定的转弯次数,问能否在规定的转弯次数内到达终点--- 这一题是学(看)习(题)的(解)@_@ 主要学了两个地方 一个是剪枝,如果搜到的当前点的转弯次数小于该点turn数 ...

  6. HDU 1728 逃离迷宫(DFS||BFS)

    逃离迷宫 Problem Description 给定一个m × n (m行, n列)的迷宫,迷宫中有两个位置,gloria想从迷宫的一个位置走到另外一个位置,当然迷宫中有些地方是空地,gloria可 ...

  7. 逃离迷宫(BFS)题解

    Problem Description 给定一个m × n (m行, n列)的迷宫,迷宫中有两个位置,gloria想从迷宫的一个位置走到另外一个位置,当然迷宫中有些地方是空地,gloria可以穿越,有 ...

  8. 2018年长沙理工大学第十三届程序设计竞赛 G 逃离迷宫 【BFS】

    链接:https://www.nowcoder.com/acm/contest/96/G 来源:牛客网 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 32768K,其他语言65536 ...

  9. HDU - 1728 逃离迷宫 【BFS】

    题目链接 http://acm.hdu.edu.cn/showproblem.php?pid=1728 思路 BFS 一开始 从开始位置 往四周走 如果能走的话 这个时候 转弯次数都是0 我们的标记不 ...

随机推荐

  1. CSS深入理解学习笔记之z-index

    1.z-index基础 z-index含义:指定了元素及其子元素的"z顺序",而"z顺序"可以决定元素的覆盖顺序.z-index值越大越在上面. z-index ...

  2. [JLOI2011]基因补全

    1973: [JLOI2011]基因补全 Time Limit: 1 Sec  Memory Limit: 256 MB Description 在生物课中我们学过,碱基组成了DNA(脱氧核糖核酸), ...

  3. nagios的安装

    Nagios通常由一个主程序(Nagios).一个插件程序(Nagios-plugins)和四个可选的ADDON(NRPE.NSCA. NSClient++和NDOUtils)组成.Nagios的监控 ...

  4. zabbix监控-自定义监控与报警(二)

    标签: linux 笔者Q:972581034 交流群:605799367.有任何疑问可与笔者或加群交流 1.web界面操用(host groups) 1.1创建一个组名字为damo 1.2配置文件解 ...

  5. 基础知识全面LINUX

    学习Linux系统的重要性应该不用多说,下面我就对Linux的基础知识进行一个全面而又简单的总结.不过建议大家还是装个Linux系统多练习,平时最好只在Linux环境下编程,这样会大有提高. linu ...

  6. JavaWeb项目架构之Kafka分布式日志队列

    架构.分布式.日志队列,标题自己都看着唬人,其实就是一个日志收集的功能,只不过中间加了一个Kafka做消息队列罢了. kafka介绍 Kafka是由Apache软件基金会开发的一个开源流处理平台,由S ...

  7. 精通libGDX游戏开发-RPG实战-欢迎来到RPG的世界

    欢迎来到RPG的世界 本章我会快速的使用tiled这样的瓷砖地图工具,来带领大家创造所设想的世界. 创建并编辑瓷砖地图 瓷砖地图(tile-based map)是广泛应用于各种游戏类型的地图格式,li ...

  8. python之闭包与装饰器

    python闭包与装饰器 闭包 在函数内部定义的函数包含对外部的作用域,而不是全局作用域名字的引用,这样的函数叫做闭包函数. 示例: #-------------------------------- ...

  9. Struts2的配置和一个简单的例子

    Struts2的配置和一个简单的例子 笔记仓库:https://github.com/nnngu/LearningNotes 简介 这篇文章主要讲如何在 IntelliJ IDEA 中使用 Strut ...

  10. 如何在Windows上安装多个MySQL

    将MySQL注册为系统服务:%MySQL_HOME%\bin>mysqld --install mysql5.1,此时,在运行中输入"services.msc"或者打开&qu ...