题目描述

【问题描述】

小 B 最近迷上了华容道,可是他总是要花很长的时间才能完成一次。于是,他想到用编程来完成华容道:给定一种局面, 华容道是否根本就无法完成,如果能完成, 最少需要多少时间。

小 B 玩的华容道与经典的华容道游戏略有不同,游戏规则是这样的:

  1. 在一个 n*m 棋盘上有 n*m 个格子,其中有且只有一个格子是空白的,其余 n*m-1个格子上每个格子上有一个棋子,每个棋子的大小都是 1*1 的;

  2. 有些棋子是固定的,有些棋子则是可以移动的;

  3. 任何与空白的格子相邻(有公共的边)的格子上的棋子都可以移动到空白格子上。

游戏的目的是把某个指定位置可以活动的棋子移动到目标位置。

给定一个棋盘,游戏可以玩 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。

输入输出样例

输入样例#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
输出样例#1:

2
-1

说明

【输入输出样例说明】

棋盘上划叉的格子是固定的,红色格子是目标位置,圆圈表示棋子,其中绿色圆圈表示目标棋子。

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

移动过程如下:

  1. 第二次游戏,空白格子的初始位置是(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 华容道的更多相关文章

  1. [NOIP2013] 提高组 洛谷P1969 积木大赛

    题目描述 春春幼儿园举办了一年一度的“积木大赛”.今年比赛的内容是搭建一座宽度为n的大厦,大厦可以看成由n块宽度为1的积木组成,第i块积木的最终高度需要是hi. 在搭建开始之前,没有任何积木(可以看成 ...

  2. [NOIP2013] 提高组 洛谷P1970 花匠

    题目描述 花匠栋栋种了一排花,每株花都有自己的高度.花儿越长越大,也越来越挤.栋栋决定 把这排中的一部分花移走,将剩下的留在原地,使得剩下的花能有空间长大,同时,栋栋希 望剩下的花排列得比较别致. 具 ...

  3. [NOIP2013] 提高组 洛谷P1967 货车运输

    题目描述 A 国有 n 座城市,编号从 1 到 n,城市之间有 m 条双向道路.每一条道路对车辆都有重量限制,简称限重.现在有 q 辆货车在运输货物, 司机们想知道每辆车在不超过车辆限重的情况下,最多 ...

  4. 洛谷 P1979 华容道 解题报告

    P1979 华容道 题目描述 小\(B\)最近迷上了华容道,可是他总是要花很长的时间才能完成一次.于是,他想到用编程来完成华容道:给定一种局面, 华容道是否根本就无法完成,如果能完成, 最少需要多少时 ...

  5. [NOIP2015] 提高组 洛谷P2615 神奇的幻方

    题目描述 幻方是一种很神奇的N*N矩阵:它由数字1,2,3,……,N*N构成,且每行.每列及两条对角线上的数字之和都相同. 当N为奇数时,我们可以通过以下方法构建一个幻方: 首先将1写在第一行的中间. ...

  6. [NOIP2014] 提高组 洛谷P2038 无线网络发射器选址

    题目描述 随着智能手机的日益普及,人们对无线网的需求日益增大.某城市决定对城市内的公共场所覆盖无线网. 假设该城市的布局为由严格平行的129 条东西向街道和129 条南北向街道所形成的网格状,并且相邻 ...

  7. [NOIP2012] 提高组 洛谷P1081 开车旅行

    题目描述 小 A 和小 B 决定利用假期外出旅行,他们将想去的城市从 1 到 N 编号,且编号较小的 城市在编号较大的城市的西边,已知各个城市的海拔高度互不相同,记城市 i 的海拔高度为 Hi,城市 ...

  8. [NOIP2012] 提高组 洛谷P1084 疫情控制

    题目描述 H 国有 n 个城市,这 n 个城市用 n-1 条双向道路相互连通构成一棵树,1 号城市是首都, 也是树中的根节点. H 国的首都爆发了一种危害性极高的传染病.当局为了控制疫情,不让疫情扩散 ...

  9. [NOIP2012] 提高组 洛谷P1080 国王游戏

    题目描述 恰逢 H 国国庆,国王邀请 n 位大臣来玩一个有奖游戏.首先,他让每个大臣在左.右 手上面分别写下一个整数,国王自己也在左.右手上各写一个整数.然后,让这 n 位大臣排 成一排,国王站在队伍 ...

随机推荐

  1. Java - 通过私有构造方法获取实例

  2. 项目实战2.3-Nginx的“远方表哥”—Tengine

    本文收录在Linux运维企业架构实战系列 今天想起当初研究nginx反向代理负载均衡时,nginx自身的upstream后端配置用着非常不舒服: 当时使用的淘宝基于nginx二次开发的Tengine, ...

  3. B - Sonya and Exhibition CodeForces - 1004B (思维题)

    B. Sonya and Exhibition time limit per test 1 second memory limit per test 256 megabytes input stand ...

  4. 排列算法汇总(下一个排列,全排列,第K个排列)

    一.下一个排列 首先,STL提供了两个用来计算排列组合关系的算法,分别是next_permutation和prev_permutation. next_permutation(nums.begin() ...

  5. 洛谷P4231 三步必杀

    题目描述: $N$ 个柱子排成一排,一开始每个柱子损伤度为0. 接下来勇仪会进行$M$ 次攻击,每次攻击可以用4个参数$l$ ,$r$ ,$s$ ,$e$ 来描述: 表示这次攻击作用范围为第$l$ 个 ...

  6. open()函数之文件操作

    #open() 文件操作 #打开文件的模式有: r,只读模式[默认] w,只写模式[不可读:不存在则创建:存在则清空内容:] x,只写模式[不可读:不存在则创建,存在则报错] a,追加模式[可读:不存 ...

  7. [原]sencha touch之panel和tabpanel

    最近在弄senchatouch的项目,所以边学习边开发,边记录,直接记录下test code如下: Panel: Ext.application({ name:'itKingApp', launch: ...

  8. 【Kubernetes】资源列表

    1.Kubernetes资源列表 https://www.cnblogs.com/linuxk/p/10436260.html

  9. JSP自定义tag控件标签

    JSP支持自定tag的方法,那就是直接讲JSP代码保存成*.tag或者*.tagx的标签定义文件.tag和tagx文件不仅支持经典jsp代码,各种标签模版代码,还支持xml样式的jsp指令代码. 按照 ...

  10. 免费生成https证书以及配置

    http升级到https需要在nginx的配置中加入证书信息,查询资料后确定生成证书两种方案   第一种:自签名证书,然后开启 CloudFlare 的 CDN 服务   //确定是否安装openss ...