/*双向bfs (得分和单项的一样多....)70*/
#include<iostream>
#include<cstdio>
#include<cstring>
#include<queue>
#define maxn 35
#define maxm 810010
using namespace std;
int n,m,q,g[maxn][maxn];
int X,Y,sx,sy,ex,ey,falg,vis[maxn][maxn];
int xx[]={,,,-};
int yy[]={,-,,};
int f[maxn][maxn][maxn][maxn],s[maxn][maxn][maxn][maxn];
struct node
{
int kx,ky,xi,yi;
node (int xxx, int yyy, int tt, int ss):kx(xxx), ky(yyy), xi(tt), yi(ss){};
};
queue<node>Q;
void Bfs()
{
if(sx==ex&&sy==ey)
{
printf("0\n");
falg=;return;
}
while(!Q.empty())Q.pop();
f[X][Y][sx][sy]=;
s[X][Y][sx][sy]=;
Q.push(node(X,Y,sx,sy));
for(int i=;i<;i++)
{
int nx=xx[i]+ex;
int ny=yy[i]+ey;
if(nx>&&nx<=n&&ny>&&ny<=m&&g[nx][ny]&&f[nx][ny][ex][ey]==)
{
f[nx][ny][ex][ey]=;
s[nx][ny][ex][ey]=;
Q.push(node(nx,ny,ex,ey));
}
}
while(!Q.empty())
{
node t=Q.front();Q.pop();
int x=t.kx,y=t.ky;
int xs=t.xi,ys=t.yi;
int T=f[x][y][xs][ys];
int k=s[x][y][xs][ys];
for(int i=;i<;i++)
{
int nx=x+xx[i];
int ny=y+yy[i];
xs=t.xi;ys=t.yi;
if(nx>&&nx<=n&&ny>&&ny<=m&&g[nx][ny])
{
if(nx==xs&&ny==ys)xs=x,ys=y;
if(f[nx][ny][xs][ys]==)
{
f[nx][ny][xs][ys]=T;
s[nx][ny][xs][ys]=k+;
Q.push(node(nx,ny,xs,ys));
}
else if(f[nx][ny][xs][ys]!=T)
{
printf("%d\n",s[nx][ny][xs][ys]+k+);
falg=;return;
}
} }
}
}
int main()
{
scanf("%d%d%d",&n,&m,&q);
for(int i=;i<=n;i++)
for(int j=;j<=m;j++)
scanf("%d",&g[i][j]);
while(q--)
{
scanf("%d%d%d%d%d%d",&X,&Y,&sx,&sy,&ex,&ey);
falg=;memset(f,,sizeof(f));Bfs();
if(falg==)printf("-1\n");
}
return ;
}
/*
后来看了别人的思路 机智啊!
本来bfs不超时的 nm*nm 但是加上q次询问就T了
问题就在这 我们重复的bfs太多了 能不能一次预处理好呢
但是考虑到每次的起点终点还有空格位置是在变化的
所以我们要与处理一些与这些无关的 但是对答案还有贡献的
考虑到目标棋子的移动必须先让空格移动到4周
所以我们把每个空格和目标棋子的相邻的组合作为状态
也就是有n*m*4个状态 每个状态不是相邻的 互相转移需要一定的步数
这个步数就使我们要预处理的东西
然后把每个状态映射成一个图 将目标棋盘转化成同样的状态 然后跑最短路
具体的细节看代码把
*/
#include<iostream>
#include<cstdio>
#include<cstring>
#include<queue>
#define maxn 35
#define maxm 810010
using namespace std;
int n,m,q,g[maxn][maxn],num,head[maxm],kx,ky,sx,sy,ex,ey;
int f[maxn][maxn],dis[maxm],vis[maxm],ans;
struct node{int u,v,t,pre;}e[maxm];
int xx[]={-,,,};//上下左右 和下面的对应
int yy[]={,,-,};
void Init()
{
scanf("%d%d%d",&n,&m,&q);
for(int i=;i<=n;i++)
for(int j=;j<=m;j++)
scanf("%d",&g[i][j]);
}
void Add(int from,int to,int dis)
{
num++;e[num].v=to;
e[num].t=dis;
e[num].pre=head[from];
head[from]=num;
}
void Bfs(int X,int Y,int xi,int yi,int k)
{
queue<int>qx,qy;
while(!qx.empty())qx.pop();
while(!qy.empty())qy.pop();
memset(f,,sizeof(f));
qx.push(xi);qy.push(yi);
f[xi][yi]=;//判重+步数记录
while(!qx.empty())
{
int x=qx.front();qx.pop();
int y=qy.front();qy.pop();
for(int i=;i<;i++)
{
int nx=x+xx[i];
int ny=y+yy[i];
if(nx==X&&ny==Y)continue;
if(nx>&&nx<=m&&ny>&&ny<=m&&g[nx][ny]&&f[nx][ny]==)
{
qx.push(nx);qy.push(ny);
f[nx][ny]=f[x][y]+;
}
}
}
if(k==)return;
for(int i=;i<;i++)
{
int nx=X+xx[i],ny=Y+yy[i];
if(f[nx][ny]==||(nx==xi&&ny==yi))continue;
Add(X**+Y*+k,X**+Y*+i,f[nx][ny]-);//一一映射
}
Add(X**+Y*+k,xi**+yi*+(k^),);//每组之间的状态连起来 这里的^ 就是左变变 上变下
}
void Ready()
{
for(int i=;i<=n;i++)
for(int j=;j<=m;j++)
{ if(!g[i][j]) continue;
if(g[i-][j]) Bfs(i,j,i-,j,);
if(g[i+][j]) Bfs(i,j,i+,j,);
if(g[i][j-]) Bfs(i,j,i,j-,);
if(g[i][j+]) Bfs(i,j,i,j+,);
}
}
void SPFA()
{
queue<int>q;memset(dis,/,sizeof(dis));
for(int i=;i<;i++)
{
int nx=sx+xx[i];
int ny=sy+yy[i];
if(nx>&&nx<=n&&ny>&&ny<=m&&g[nx][ny]&&f[nx][ny])
{
int k=sx**+sy*+i;
q.push(k);vis[k]=;dis[k]=f[nx][ny]-;//转化成可表示的状态
}
}
while(!q.empty())
{
int k=q.front();q.pop();vis[k]=;
for(int i=head[k];i;i=e[i].pre)
if(dis[e[i].v]>dis[k]+e[i].t)
{
dis[e[i].v]=dis[k]+e[i].t;
if(vis[e[i].v]==)
{
vis[e[i].v]=;q.push(e[i].v);
}
}
}
}
int main()
{
Init();
Ready();
for(int i=;i<=q;i++)
{
scanf("%d%d%d%d%d%d",&kx,&ky,&sx,&sy,&ex,&ey);
if(sx==ex&&sy==ey)
{
printf("0\n");continue;
}
Bfs(sx,sy,kx,ky,);//把空格移到目标棋子附近
SPFA();ans=dis[];
for(int i=;i<;i++)
{
int k=ex**+ey*+i;
ans=min(ans,dis[k]);
}
if(ans==dis[])printf("-1\n");
else printf("%d\n",ans);
}
return ;
}

noip 2013 华容道的更多相关文章

  1. Luogu 1979 NOIP 2013 华容道(搜索,最短路径)

    Luogu 1979 NOIP 2013 华容道(搜索,最短路径) Description 小 B 最近迷上了华容道,可是他总是要花很长的时间才能完成一次.于是,他想到用编程来完成华容道:给定一种局面 ...

  2. 洛谷 P1979 [ NOIP 2013 ] 华容道 —— bfs + 最短路

    题目:https://www.luogu.org/problemnew/show/P1979 真是一道好题... 首先考虑暴力做法,应该是设 f[i][j][x][y] 记录指定棋子和空格的位置,然后 ...

  3. NOIP 2013 货车运输【Kruskal + 树链剖分 + 线段树 】【倍增】

    NOIP 2013 货车运输[树链剖分] 树链剖分 题目描述 Description A 国有 n 座城市,编号从 1 到 n,城市之间有 m 条双向道路.每一条道路对车辆都有重量限制,简称限重.现在 ...

  4. [Noip 2013 Day1-3] 货车运输 做法总结

    [Noip 2013 Day1-3] 货车运输 做法总结 Online Judge:Luogu-1967 Label:启发式合并,离线,整体二分,按秩合并,倍增,最大生成树 打模拟离线赛时做到,顺便总 ...

  5. 【CodeVS 3290】【NOIP 2013】华容道

    http://codevs.cn/problem/3290/ 据说2013年的noip非常难,但Purpleslz学长还是AK了.能A掉这道题真心orz. 设状态$(i,j,k)$表示目标棋子在$(i ...

  6. 【NOIP 2013 DAY2 T3】 华容道(spfa)

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

  7. 华容道 [NOIP 2013]

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

  8. 【noip】华容道

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

  9. NOIP 2013

    Prob.1 转圈游戏 找到循环节,然后快速幂.代码: #include<cstdio> #include<cstring> #include<iostream> ...

随机推荐

  1. windows server 2008 asp连接数据库sql2000失败

    由于服务器现在的服务器已不能承受了,需要替换服务器并把window2003升级为window2008,把所有数据都平移过来.平移完后遇到ASP总是不能连接上sql2000,这让我非常郁闷,处理了好几个 ...

  2. Mac iOS-----PhoneGap 添加条形扫描码插件

    此随笔记录的是通过plugman来安装phonegap插件. 第1步:安装plugman npm install -g plugman QiaodeMacBook-Pro:MyApp root# np ...

  3. Phalcon的学习篇-phalcon和devtools的安装和设置

    A Phalcon在Windows上的安装 1 从Phalcon for Windows下载适合的DLL, 这里的适合 主要看两个方面 1 PHP的版本 2 线程是否是安全 3 编译版本 如果不清楚这 ...

  4. C++类:private、public、friend、protected的区别

           private和public的作用是让编译器帮你检查某些模块是否使用了他没权限使用的模块,也就是生成可执行代码的时候做权限检查.比如,公司里各个部门有自己私有的信息,财务部可以看所有员工 ...

  5. [BZOJ 1692] [Usaco2007 Dec] 队列变换 【后缀数组 + 贪心】

    ---恢复内容开始--- 题目链接:BZOJ - 1692 题目分析 首先,有个比较简单的贪心思路:如果当前剩余字符串的两端字母不同,就选取小的字母,这样显然是正确的. 然而若两端字母相同,我们怎么选 ...

  6. 查看SharePoint list的xml

    http://{0}/_vti_bin/owssvr.dll?Cmd=Display&List={1}&XMLDATA=TRUE {0} – The URL to your site. ...

  7. Python4Delphi也是与VCL密切相关,所以才能相互调用,绝对有研究价值!

    Python4Delphi也是与VCL密切相关,所以才能相互调用,绝对有研究价值! http://www.cnblogs.com/GarfieldTom/archive/2013/01/17/2864 ...

  8. SORT ORDER BY STOPKEY

    select * from ( select * from ( select a.*,rownum rn from page a where object_id >1000 and owner= ...

  9. 实战weblogic集群之创建domain,AdminServer

    在weblogic安装完后,接下来就可以创建domain,AdminSever了. 第1步: $ cd /app/sinova/Oracle/wlserver_10./common/bin $ ./c ...

  10. 数学(线性规划): ZJOI2013 防守战线

    偷懒用的线性规划. #include <iostream> #include <cstring> #include <cstdio> using namespace ...