题目描述

【问题描述】

小 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:

  1. 3 4 2
  2. 0 1 1 1
  3. 0 1 1 0
  4. 0 1 0 0
  5. 3 2 1 2 2 2
  6. 1 2 2 2 3 2
输出样例#1:

  1. 2
  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方向时的状态,然后预处理出每个状态之间转移的代价,根据转移方向和代价建立图,求最短路即可。

指定块向某方向移动时,需要将空白块先移动到目标方向,然后使两者交换位置。据此可以预处理方块在每个位置向每个方向移动的代价。转移前后的状态,第三维表示的“方向”相反,写特判写得心灰意懒,之后在别的题解学到了异或大法,具体看代码 (异或大法好!)。

——此外还要记得判断无法到达和起终点相同的情况。

  1. /**/
  2. #include<iostream>
  3. #include<cstdio>
  4. #include<cmath>
  5. #include<cstring>
  6. #include<algorithm>
  7. #include<queue>
  8. using namespace std;
  9. const int INF=0x3f3f3f3f;
  10. const int mxn=;
  11. //bas
  12. int n,m,c;
  13. int ex,ey,sx,sy,tx,ty;
  14. //edge
  15. struct edge{
  16. int v,dis;
  17. int next;
  18. }e[];
  19. int hd[],cnt=;
  20. void add_edge(int u,int v,int dis){
  21. e[++cnt].next=hd[u];e[cnt].v=v;e[cnt].dis=dis;hd[u]=cnt;
  22. return;
  23. }
  24. //map
  25. int d[mxn][mxn][][];
  26. int id[mxn][mxn][];
  27. int mp[mxn][mxn];
  28. int dis[mxn][mxn];
  29. int mx[]={,-,,},//上,下,右,左
  30. my[]={,,,-};
  31. //edge_check
  32. inline bool pd(int x,int y){
  33. if(x> && x<=n && y> && y<=m)return ;
  34. return ;
  35. }
  36. queue< pair<int,int> >q;
  37. int BFS(int x1,int y1,int x2,int y2){
  38. if(x1==x2 && y1==y2)return ;
  39. memset(dis,,sizeof dis);
  40. while(!q.empty()) q.pop();
  41. q.push(make_pair(x1,y1));
  42. while(!q.empty()){
  43. int x=q.front().first;
  44. int y=q.front().second;
  45. for(int i=;i<;i++){
  46. int nx=x+mx[i],ny=y+my[i];
  47. if(!pd(nx,ny))continue;
  48. if(!mp[nx][ny])continue;
  49. if(dis[nx][ny])continue;
  50. dis[nx][ny]=dis[x][y]+;
  51. if(nx==x2 && ny==y2)return dis[nx][ny];
  52. q.push(make_pair(nx,ny));
  53. }
  54. q.pop();
  55. }
  56. return INF;
  57. }
  58. int dist[];
  59. bool inq[];
  60. queue<int>qu;
  61. int SPFA(int S,int T){
  62. memset(dist,0x3f,sizeof dist);
  63. memset(inq,,sizeof inq);
  64. while(!qu.empty()) qu.pop();
  65. qu.push(S);
  66. dist[S]=;
  67. inq[S]=;
  68. while(!qu.empty()){
  69. int now=qu.front();
  70. qu.pop();
  71. // printf("u:%d\n",now);
  72. for(int i=hd[now];i;i=e[i].next){
  73. int v=e[i].v;
  74. if(e[i].dis+dist[now]<dist[v]){
  75. // printf("-- %d\n",dist[now]+e[i].dis);
  76. dist[v]=dist[now]+e[i].dis;
  77. if(!inq[v]){
  78. inq[v]=;
  79. qu.push(v);
  80. }
  81. }
  82. }
  83. inq[now]=;
  84. }
  85. if(dist[T]==INF)return -;
  86. return dist[T];
  87. }
  88.  
  89. int mcnt=;
  90. void init(){
  91. memset(hd,,sizeof hd);
  92. int i,j,k;
  93. for(i=;i<=n;i++)
  94. for(j=;j<=m;j++)
  95. for(k=;k<;k++)
  96. id[i][j][k]=++mcnt;
  97. //以下部分预处理出对于一个格子(i,j),把空格从它的k1方向移动到k2方向需要的步数
  98. memset(d,0x3f,sizeof d);
  99. for(i=;i<=n;i++)
  100. for(j=;j<=m;j++)
  101. if(mp[i][j]){
  102. mp[i][j]=;
  103. for(int k1=;k1<;k1++){
  104. int nx=i+mx[k1],ny=j+my[k1];
  105. if(!pd(nx,ny))continue;
  106. if(!mp[nx][ny])continue;
  107. for(int k2=;k2<;k2++){
  108. int ax=i+mx[k2],ay=j+my[k2];
  109. if(!pd(ax,ay))continue;
  110. if(mp[ax][ay])d[i][j][k1][k2]=BFS(nx,ny,ax,ay)+;
  111. }
  112. }
  113. mp[i][j]=;
  114. }
  115. //预处理当空白格子在(x,y)的k1方向时,把格子(x,y)向k2方向移动一格需要的步数
  116. for(i=;i<=n;i++)
  117. for(j=;j<=m;j++)
  118. for(int k1=;k1<;k1++)
  119. for(int k2=;k2<;k2++)
  120. if(d[i][j][k1][k2]!=INF)
  121. add_edge(id[i][j][k1],id[i+mx[k2]][j+my[k2]][k2^],d[i][j][k1][k2]);
  122. return;
  123. }
  124. int main(){
  125. scanf("%d%d%d",&n,&m,&c);
  126. int i,j;
  127. for(i=;i<=n;i++)
  128. for(j=;j<=m;j++)
  129. scanf("%d",&mp[i][j]);
  130. init();
  131. while(c--){
  132. scanf("%d%d%d%d%d%d",&ex,&ey,&sx,&sy,&tx,&ty);
  133. if(!pd(ex,ey) || !pd(sx,sy) || !pd(tx,ty) ||
  134. !mp[sx][sy] || !mp[tx][ty] || !mp[ex][ey] ){
  135. printf("-1\n");
  136. continue;
  137. }
  138. if(sx==tx && sy==ty){
  139. printf("0\n");
  140. continue;
  141. }
  142. int S=++mcnt,T=++mcnt;
  143. mp[sx][sy]=;
  144. for(int k=;k<;k++){
  145. int nx=sx+mx[k],ny=sy+my[k];
  146. if(mp[nx][ny]){
  147. int tmp=BFS(ex,ey,nx,ny);//把空格移动到起点周边的步数
  148. if(tmp!=INF)add_edge(S,id[sx][sy][k],tmp);
  149. }
  150. }
  151. mp[sx][sy]=;
  152. for(int k=;k<;k++){
  153. int nx=tx+mx[k],ny=ty+my[k];
  154. if(mp[nx][ny]) add_edge(id[tx][ty][k],T,);//建从终点周围到终点的边
  155. }
  156. printf("%d\n",SPFA(S,T));
  157. }
  158. return ;
  159. }

[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. cf550D. Regular Bridge(构造)

    题意 给出一个$k$,构造一个无向图,使得每个点的度数为$k$,且存在一个桥 Sol 神仙题 一篇写的非常好的博客:http://www.cnblogs.com/mangoyang/p/9302269 ...

  2. PAT 乙级 1059

    题目 题目地址:PAT 乙级 1059 题解 开始我是从暴力循环的角度考虑这道题,大概计算了一下时间复杂度应该不会超,但是很不幸没有通过,时间超限:之后考虑搜索算法可能优化不太好,因此就把输入的序列先 ...

  3. JavaScript设置div中的文字滚动起来 实现滚动效果

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  4. http 工作模式与模块

    目录 http 工作模式与模块 http 服务器应用 MPM工作模式 prefork worker event 进程角色 httpd功能特性 http 安装 centos6配置目录 http 2.2 ...

  5. windows 时间同步至最新时间方法 | windows 时间同步服务器

    国内 windows 系统的电脑有时候不能自动同步互联网当前时间,这就需要改一下 windows 的时间同步服务器 版权声明:本文为博主原创文章,未经博主允许不得转载. 原文地址:https://ww ...

  6. Vuejs中关于computed、methods、watch的区别。

    Vue.js在模板表达式中限制了,绑定表达式最多只能有一条表达式,但某些数据需要一条以上的表达式运算实现,此时就可以将此数据放在计算属性(computed)当中. Vuejs中关于computed.m ...

  7. spring boot 设置tomcat post参数限制

    今天传图片,用的base64字符串,POST方法,前端传送的时候总是莫名其妙的崩溃,去网上搜了半天,以为是文件大小被限制了,但是我这个是字符串接收,不是文件接收,于是又继续搜,原来post本身没有参数 ...

  8. windows7下将Cygwin加入右键菜单,并从当前目录打开

    第一步:修改windows注册表 1·开始->运行(或者win键+R),输入REGEDIT,回车,打开注册表编辑器: 2·找到HKEY_CLASSES_ROOT\Directory\Backgr ...

  9. DFS:Prime Ring Problem(素数环)

    解体心得: 1.一个回溯法,可以参考八皇后问题. 2.题目要求按照字典序输出,其实在按照回溯法得到的答案是很正常的字典序.不用去特意排序. 3.输出有个坑,就是在输出一串的最后不能有空格,不然要PE, ...

  10. 菜鸟学Linux - 文件/文件夹的隐藏属性

    文件/文件夹居然还有隐藏属性?没错,隐藏属性对于文件/文件夹的安全很重要.好比如说,我们需要使用”鉴定符“来揭开装备的隐藏属性:在Linux中chattr/lsattr就是“鉴定符”. chattr基 ...