/*双向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. [Python]更加Pythonic的多个List合并和Python的安利

    原题: https://segmentfault.com/q/1010000005904259 问题: 倘若存在 L=[ [1,2,3],[4,5,6],[7,8,9]] 这样的列表,如何把合并成[1 ...

  2. JS Math.sin() 与 Math.cos() 用法

    Math.sin(x)      x 的正玄值.返回值在 -1.0 到 1.0 之间: Math.cos(x)    x 的余弦值.返回的是 -1.0 到 1.0 之间的数: 这两个函数中的X 都是指 ...

  3. MapReduce程序开发之流量求和(八)

    1.分析记录手机流量的日志. 2.拿到日志中的一行数据,切分各个字段,抽取出我们需要的字段:手机号,上行流量,下行流量,然后封装成kv发送出去 3.使用java中的map方法: public clas ...

  4. winform布局格式

    一.默认布局 ★可以加panel,也可以不加: ★通过鼠标拖动控件的方式,根据自己的想法布局.拖动控件的过程中,会有对齐的线,方便操作: ★也可选中要布局的控件,在工具栏中有对齐工具可供选择,也有调整 ...

  5. JQuery重要知识点

    jQuery基本选择器----包括ID选择器,标签选择器,类选择器,通配选择器和组选择器5种 a. ID选择器: $("#id") b. 标签选择器:$("element ...

  6. poj2228

    这显然是一道环形dp的题目 处理环形我们都是要转化为线性来做 一般有这么两种方法处理 复制一段到最后 (比如说noip的能量项链) 考查环形对dp的影响然后分类讨论(比如bzoj1040) 这道题我们 ...

  7. SQL约束和字段约束的创建和删除

    1)禁止所有表约束的SQLselect 'alter table '+name+' nocheck constraint all' from sysobjects where type='U' 2)删 ...

  8. 【最小生成树】Codeforces 707B Bakery

    题目链接: http://codeforces.com/problemset/problem/707/B 题目大意: 给你N个点M条无向边,其中有K个面粉站,现在一个人要在不是面粉站的点上开店,问到面 ...

  9. 控件treeview使用

    一:实现功能,获得选中节点,在选中节点下添加节点,折叠,展开,删除,得到选中节点下checked项,选中根节点其下节点也选中,图标.上图 二:相关代码 using System; using Syst ...

  10. HDU1247 Hat’s Words(Trie的动态内存版本)

    Hat’s Words Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Tota ...