[NOIP2013] 提高组 洛谷P1979 华容道
题目描述
【问题描述】
小 B 最近迷上了华容道,可是他总是要花很长的时间才能完成一次。于是,他想到用编程来完成华容道:给定一种局面, 华容道是否根本就无法完成,如果能完成, 最少需要多少时间。
小 B 玩的华容道与经典的华容道游戏略有不同,游戏规则是这样的:
在一个 n*m 棋盘上有 n*m 个格子,其中有且只有一个格子是空白的,其余 n*m-1个格子上每个格子上有一个棋子,每个棋子的大小都是 1*1 的;
有些棋子是固定的,有些棋子则是可以移动的;
- 任何与空白的格子相邻(有公共的边)的格子上的棋子都可以移动到空白格子上。
游戏的目的是把某个指定位置可以活动的棋子移动到目标位置。
给定一个棋盘,游戏可以玩 q 次,当然,每次棋盘上固定的格子是不会变的, 但是棋盘上空白的格子的初始位置、 指定的可移动的棋子的初始位置和目标位置却可能不同。第 i 次
玩的时候, 空白的格子在第 EXi 行第 EYi 列,指定的可移动棋子的初始位置为第 SXi 行第 SYi列,目标位置为第 TXi 行第 TYi 列。
假设小 B 每秒钟能进行一次移动棋子的操作,而其他操作的时间都可以忽略不计。请你告诉小 B 每一次游戏所需要的最少时间,或者告诉他不可能完成游戏。
输入输出格式
输入格式:
输入文件为 puzzle.in。
第一行有 3 个整数,每两个整数之间用一个空格隔开,依次表示 n、m 和 q;
接下来的 n 行描述一个 n*m 的棋盘,每行有 m 个整数,每两个整数之间用一个空格隔开,每个整数描述棋盘上一个格子的状态,0
表示该格子上的棋子是固定的,1 表示该格子上的棋子可以移动或者该格子是空白的。接下来的 q 行,每行包含 6 个整数依次是
EXi、EYi、SXi、SYi、TXi、TYi,每两个整数之间用一个空格隔开,表示每次游戏空白格子的位置,指定棋子的初始位置和目标位置。
输出格式:
输出文件名为 puzzle.out。
输出有 q 行,每行包含 1 个整数,表示每次游戏所需要的最少时间,如果某次游戏无法完成目标则输出−1。
输入输出样例
3 4 2
0 1 1 1
0 1 1 0
0 1 0 0
3 2 1 2 2 2
1 2 2 2 3 2
2
-1
说明
【输入输出样例说明】
棋盘上划叉的格子是固定的,红色格子是目标位置,圆圈表示棋子,其中绿色圆圈表示目标棋子。
- 第一次游戏,空白格子的初始位置是 (3, 2)(图中空白所示),游戏的目标是将初始位置在(1, 2)上的棋子(图中绿色圆圈所代表的棋子)移动到目标位置(2, 2)(图中红色的格子)上。
移动过程如下:

- 第二次游戏,空白格子的初始位置是(1, 2)(图中空白所示),游戏的目标是将初始位置在(2, 2)上的棋子(图中绿色圆圈所示)移动到目标位置 (3, 2)上。

要将指定块移入目标位置,必须先将空白块移入目标位置,空白块要移动到目标位置,必然是从位置(2, 2)上与当前图中目标位置上的棋子交换位置,之后能与空白块交换位置的只有当前图中目标位置上的那个棋子,因此目标棋子永远无法走到它的目标位置, 游戏无
法完成。
【数据范围】
对于 30%的数据,1 ≤ n, m ≤ 10,q = 1;
对于 60%的数据,1 ≤ n, m ≤ 30,q ≤ 10;
对于 100%的数据,1 ≤ n, m ≤ 30,q ≤ 500。
比较麻烦的搜索题。用三维数组 [i][j][k]表示指定块在(i,j)位置,空格在它的k方向时的状态,然后预处理出每个状态之间转移的代价,根据转移方向和代价建立图,求最短路即可。
指定块向某方向移动时,需要将空白块先移动到目标方向,然后使两者交换位置。据此可以预处理方块在每个位置向每个方向移动的代价。转移前后的状态,第三维表示的“方向”相反,写特判写得心灰意懒,之后在别的题解学到了异或大法,具体看代码 (异或大法好!)。
——此外还要记得判断无法到达和起终点相同的情况。
/**/
#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<algorithm>
#include<queue>
using namespace std;
const int INF=0x3f3f3f3f;
const int mxn=;
//bas
int n,m,c;
int ex,ey,sx,sy,tx,ty;
//edge
struct edge{
int v,dis;
int next;
}e[];
int hd[],cnt=;
void add_edge(int u,int v,int dis){
e[++cnt].next=hd[u];e[cnt].v=v;e[cnt].dis=dis;hd[u]=cnt;
return;
}
//map
int d[mxn][mxn][][];
int id[mxn][mxn][];
int mp[mxn][mxn];
int dis[mxn][mxn];
int mx[]={,-,,},//上,下,右,左
my[]={,,,-};
//edge_check
inline bool pd(int x,int y){
if(x> && x<=n && y> && y<=m)return ;
return ;
}
queue< pair<int,int> >q;
int BFS(int x1,int y1,int x2,int y2){
if(x1==x2 && y1==y2)return ;
memset(dis,,sizeof dis);
while(!q.empty()) q.pop();
q.push(make_pair(x1,y1));
while(!q.empty()){
int x=q.front().first;
int y=q.front().second;
for(int i=;i<;i++){
int nx=x+mx[i],ny=y+my[i];
if(!pd(nx,ny))continue;
if(!mp[nx][ny])continue;
if(dis[nx][ny])continue;
dis[nx][ny]=dis[x][y]+;
if(nx==x2 && ny==y2)return dis[nx][ny];
q.push(make_pair(nx,ny));
}
q.pop();
}
return INF;
}
int dist[];
bool inq[];
queue<int>qu;
int SPFA(int S,int T){
memset(dist,0x3f,sizeof dist);
memset(inq,,sizeof inq);
while(!qu.empty()) qu.pop();
qu.push(S);
dist[S]=;
inq[S]=;
while(!qu.empty()){
int now=qu.front();
qu.pop();
// printf("u:%d\n",now);
for(int i=hd[now];i;i=e[i].next){
int v=e[i].v;
if(e[i].dis+dist[now]<dist[v]){
// printf("-- %d\n",dist[now]+e[i].dis);
dist[v]=dist[now]+e[i].dis;
if(!inq[v]){
inq[v]=;
qu.push(v);
}
}
}
inq[now]=;
}
if(dist[T]==INF)return -;
return dist[T];
} int mcnt=;
void init(){
memset(hd,,sizeof hd);
int i,j,k;
for(i=;i<=n;i++)
for(j=;j<=m;j++)
for(k=;k<;k++)
id[i][j][k]=++mcnt;
//以下部分预处理出对于一个格子(i,j),把空格从它的k1方向移动到k2方向需要的步数
memset(d,0x3f,sizeof d);
for(i=;i<=n;i++)
for(j=;j<=m;j++)
if(mp[i][j]){
mp[i][j]=;
for(int k1=;k1<;k1++){
int nx=i+mx[k1],ny=j+my[k1];
if(!pd(nx,ny))continue;
if(!mp[nx][ny])continue;
for(int k2=;k2<;k2++){
int ax=i+mx[k2],ay=j+my[k2];
if(!pd(ax,ay))continue;
if(mp[ax][ay])d[i][j][k1][k2]=BFS(nx,ny,ax,ay)+;
}
}
mp[i][j]=;
}
//预处理当空白格子在(x,y)的k1方向时,把格子(x,y)向k2方向移动一格需要的步数
for(i=;i<=n;i++)
for(j=;j<=m;j++)
for(int k1=;k1<;k1++)
for(int k2=;k2<;k2++)
if(d[i][j][k1][k2]!=INF)
add_edge(id[i][j][k1],id[i+mx[k2]][j+my[k2]][k2^],d[i][j][k1][k2]);
return;
}
int main(){
scanf("%d%d%d",&n,&m,&c);
int i,j;
for(i=;i<=n;i++)
for(j=;j<=m;j++)
scanf("%d",&mp[i][j]);
init();
while(c--){
scanf("%d%d%d%d%d%d",&ex,&ey,&sx,&sy,&tx,&ty);
if(!pd(ex,ey) || !pd(sx,sy) || !pd(tx,ty) ||
!mp[sx][sy] || !mp[tx][ty] || !mp[ex][ey] ){
printf("-1\n");
continue;
}
if(sx==tx && sy==ty){
printf("0\n");
continue;
}
int S=++mcnt,T=++mcnt;
mp[sx][sy]=;
for(int k=;k<;k++){
int nx=sx+mx[k],ny=sy+my[k];
if(mp[nx][ny]){
int tmp=BFS(ex,ey,nx,ny);//把空格移动到起点周边的步数
if(tmp!=INF)add_edge(S,id[sx][sy][k],tmp);
}
}
mp[sx][sy]=;
for(int k=;k<;k++){
int nx=tx+mx[k],ny=ty+my[k];
if(mp[nx][ny]) add_edge(id[tx][ty][k],T,);//建从终点周围到终点的边
}
printf("%d\n",SPFA(S,T));
}
return ;
}
[NOIP2013] 提高组 洛谷P1979 华容道的更多相关文章
- [NOIP2013] 提高组 洛谷P1969 积木大赛
题目描述 春春幼儿园举办了一年一度的“积木大赛”.今年比赛的内容是搭建一座宽度为n的大厦,大厦可以看成由n块宽度为1的积木组成,第i块积木的最终高度需要是hi. 在搭建开始之前,没有任何积木(可以看成 ...
- [NOIP2013] 提高组 洛谷P1970 花匠
题目描述 花匠栋栋种了一排花,每株花都有自己的高度.花儿越长越大,也越来越挤.栋栋决定 把这排中的一部分花移走,将剩下的留在原地,使得剩下的花能有空间长大,同时,栋栋希 望剩下的花排列得比较别致. 具 ...
- [NOIP2013] 提高组 洛谷P1967 货车运输
题目描述 A 国有 n 座城市,编号从 1 到 n,城市之间有 m 条双向道路.每一条道路对车辆都有重量限制,简称限重.现在有 q 辆货车在运输货物, 司机们想知道每辆车在不超过车辆限重的情况下,最多 ...
- 洛谷 P1979 华容道 解题报告
P1979 华容道 题目描述 小\(B\)最近迷上了华容道,可是他总是要花很长的时间才能完成一次.于是,他想到用编程来完成华容道:给定一种局面, 华容道是否根本就无法完成,如果能完成, 最少需要多少时 ...
- [NOIP2015] 提高组 洛谷P2615 神奇的幻方
题目描述 幻方是一种很神奇的N*N矩阵:它由数字1,2,3,……,N*N构成,且每行.每列及两条对角线上的数字之和都相同. 当N为奇数时,我们可以通过以下方法构建一个幻方: 首先将1写在第一行的中间. ...
- [NOIP2014] 提高组 洛谷P2038 无线网络发射器选址
题目描述 随着智能手机的日益普及,人们对无线网的需求日益增大.某城市决定对城市内的公共场所覆盖无线网. 假设该城市的布局为由严格平行的129 条东西向街道和129 条南北向街道所形成的网格状,并且相邻 ...
- [NOIP2012] 提高组 洛谷P1081 开车旅行
题目描述 小 A 和小 B 决定利用假期外出旅行,他们将想去的城市从 1 到 N 编号,且编号较小的 城市在编号较大的城市的西边,已知各个城市的海拔高度互不相同,记城市 i 的海拔高度为 Hi,城市 ...
- [NOIP2012] 提高组 洛谷P1084 疫情控制
题目描述 H 国有 n 个城市,这 n 个城市用 n-1 条双向道路相互连通构成一棵树,1 号城市是首都, 也是树中的根节点. H 国的首都爆发了一种危害性极高的传染病.当局为了控制疫情,不让疫情扩散 ...
- [NOIP2012] 提高组 洛谷P1080 国王游戏
题目描述 恰逢 H 国国庆,国王邀请 n 位大臣来玩一个有奖游戏.首先,他让每个大臣在左.右 手上面分别写下一个整数,国王自己也在左.右手上各写一个整数.然后,让这 n 位大臣排 成一排,国王站在队伍 ...
随机推荐
- SummerVocation_Learning--java的线程同步
public class Test_XCTB implements Runnable{ Timer timer = new Timer(); public static void main(Strin ...
- Linux运维常用命令详解
1.ls 文件属性: -:普通文件 d:目录文件 b:块设备 c:字符设备文件 l:符号连接文件 p:命令管道 s:套接字文件 文件权限: 9位数字,每3位一组 文件硬链接次数 文 ...
- 汇编 if else 例子
.text .global _start _start: mov r1,#1 mov r2,#2 cmp r1, r2 bgt branch1 add r3,r1,r2 b end branc ...
- 数据追踪系统Zipkin 及其 Zipkin的php客户端驱动hoopak
Zipkin是Twitter的一个开源项目,是一个致力于收集Twitter所有服务的监控数据的分布式跟踪系统,它提供了收集数据,和查询数据两大接口服务.Zipkin 是一款开源的分布式实时数据追踪系统 ...
- php+croppic.js实现剪切上传图片
最近需要实现裁剪图片上传,想起之前公司用到的一个插件,却不知道叫什么名字了. 在网上找了有些时间,最终找到了这个网站. http://www.croppic.net/ 因为官网文档全部都是英文,所以看 ...
- JavaScript事件对象与事件的委托
事件对象 包含事件相关的信息,如鼠标.时间.触发的DOM对象等 js默认将事件对象封装好,并自动的以参数的形式,传递给事件处理函数的第1个参数,如下: document.getElementsByTa ...
- 第9课 文章模块分析及建表 Thinkphp5商城第四季
目录 文章模块的分析 表结构 文章模块的分析 表结构 CREATE TABLE `tp_cate` ( `id` smallint(6) NOT NULL AUTO_INCREMENT COMMENT ...
- 理解 Objective-c "属性"
理解 Objective-c "属性" @property 是OC中能够快速定义一个属性的关键字,如下我们定义一个属性. @property NSString *String; 这 ...
- Keywords Search HDU - 2222 ( ac自动机)模版题
Keywords Search Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 131072/131072 K (Java/Others ...
- Diycode开源项目 NodeListFragment分析
1.整体分析 1.1.作用讲解 这个类没有用到,只是一个备用的. 分类列表,用于定制首页内容 考虑到节点列表一般不会变化,所以直接将数据封装在客户端中,可以直接查看,提高效率. 1.2.整体分析 首先 ...