luogu1979 华容道 (dijkstra+bfs)

Posted Ressed.

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了luogu1979 华容道 (dijkstra+bfs)相关的知识,希望对你有一定的参考价值。

我想动某个点的话,一定要先把空白点移动到这个点旁边,然后调换这个点和空白点,一直重复

那么,我们就可以记一些状态(x,y,s) (s={0,1},{0,-1},{1,0},{-1,0}),表示我要动的点在(x,y),然后空白点在(x+s.x,y+s.y)

这样的话我们就可以建图:$(x,y,s)-1->(x+s.x,y+s.y,s^{-1})$ (s^{-1}表示一个相反的方向,比如如果s是右边的话,那它就是左边)

$(x,y,s)-C(x,y,s+s.x,y+s.y,x+i.x,y+i.y)->(x,y,i)$ 其中,C(a,b,c,d,e,f)表示我目前这个点在(a,b),空白点在(c,d),要把空白点挪到(e,f)的最小步数,而且挪的过程中不能经过(a,b)(要不然会把要动的那个点换走)

这样的话,就可以一边bfs算出C的值,一边跑最短路了。

然而复杂度$O(q*(mn)^2)$,过不去。

考虑预处理出C,直接做显然不行($mn^3$),但其实我们要用到C的情况,都是空白点在要处理的点的旁边的,那它就可以压缩成(x,y,s,tx,ty),是900*4*900的。

当然,一开始我们的空白点和要做的点并没有在一起,只要再单独bfs一下,先把空白点挪到它边上就可以了。

要注意有起始点和目标点相同的情况,要判掉。

 1 #include<bits/stdc++.h>
 2 #define pa pair<int,int>
 3 #define pa2 pair<int,Node>
 4 #define ll long long
 5 using namespace std;
 6 const int maxn=33;
 7 
 8 inline ll rd(){
 9     ll x=0;char c=getchar();int neg=1;
10     while(c<0||c>9){if(c==-) neg=-1;c=getchar();}
11     while(c>=0&&c<=9) x=x*10+c-0,c=getchar();
12     return x*neg;
13 }
14 
15 struct Node{
16     int x,y,s;
17     Node (int x1,int x2,int x3){x=x1,y=x2,s=x3;}
18 };
19 int f[maxn][maxn],N,M,Q,dis[maxn][maxn][4][maxn][maxn],dis2[maxn][maxn][4];
20 int step[4][2]={{0,1},{0,-1},{1,0},{-1,0}};
21 bool can[maxn][maxn],flag[maxn][maxn],flag2[maxn][maxn][4];
22 queue<pa > q;
23 priority_queue<pa2,vector<pa2>,greater<pa2> > q2;
24 
25 bool operator < (Node a,Node b){return a.x<b.x;}
26 
27 int bfs1(int a,int b,int c,int x0,int y0){
28     memset(flag,0,sizeof(flag));
29     while(!q.empty()) q.pop();q.push(make_pair(x0,y0));
30     dis[a][b][c][x0][y0]=0;
31     while(!q.empty()){
32         int x=q.front().first,y=q.front().second;q.pop();
33         for(int i=0;i<=3;i++){
34             int xx=x+step[i][0],yy=y+step[i][1];
35             if(!flag[xx][yy]&&can[xx][yy]){
36                 q.push(make_pair(xx,yy));
37                 dis[a][b][c][xx][yy]=dis[a][b][c][x][y]+1;
38             }
39         }flag[x][y]=1;
40     }
41 }
42 
43 int dijkstra(int ex,int ey,int sx,int sy,int tx,int ty){
44     if(sx==tx&&sy==ty) return 0;
45     memset(dis2,-1,sizeof(dis2));memset(flag2,0,sizeof(flag2));
46     while(!q2.empty()) q2.pop();
47     can[sx][sy]=0;memset(dis[0][0][0],-1,sizeof(dis[0][0][0]));
48     bfs1(0,0,0,ex,ey);
49     for(int i=0;i<=3;i++){
50         int xx=sx+step[i][0],yy=sy+step[i][1];
51         if(dis[0][0][0][xx][yy]!=-1){
52             dis2[xx][yy][i^1]=dis[0][0][0][xx][yy]+1;
53             q2.push(make_pair(dis2[xx][yy][i^1],Node(xx,yy,i^1)));
54         }
55     }can[sx][sy]=1;
56     while(!q2.empty()){
57         Node p=q2.top().second;q2.pop();if(flag2[p.x][p.y][p.s]) continue;
58         for(int i=0;i<=3;i++){
59             int xx=p.x+step[i][0],yy=p.y+step[i][1];
60             if(dis[p.x][p.y][p.s][xx][yy]!=-1){
61                 if(dis2[xx][yy][i^1]==-1||dis2[xx][yy][i^1]>dis[p.x][p.y][p.s][xx][yy]+dis2[p.x][p.y][p.s]+1){
62                     dis2[xx][yy][i^1]=dis[p.x][p.y][p.s][xx][yy]+dis2[p.x][p.y][p.s]+1;
63                     q2.push(make_pair(dis2[xx][yy][i^1],Node(xx,yy,i^1)));
64                 }
65             }
66         }flag2[p.x][p.y][p.s]=1;
67     }int re=-1;
68     for(int i=0;i<=3;i++){
69         if(dis2[tx][ty][i]==-1) continue;
70         if(re==-1) re=dis2[tx][ty][i];
71         else re=min(re,dis2[tx][ty][i]);
72     }return re;
73 }
74 
75 int main(){
76     int i,j,k;
77     N=rd(),M=rd();Q=rd();
78     for(i=1;i<=N;i++){
79         for(j=1;j<=M;j++) can[i][j]=rd();
80     }
81     memset(dis,-1,sizeof(dis));
82     for(i=1;i<=N;i++){
83         for(j=1;j<=M;j++){
84             if(!can[i][j]) continue;can[i][j]=0;
85             for(k=0;k<=3;k++){
86                 int ii=i+step[k][0],jj=j+step[k][1];
87                 if(can[ii][jj]) bfs1(i,j,k,ii,jj);
88             }can[i][j]=1;
89         }
90     }
91     for(i=1;i<=Q;i++){
92         int x1=rd(),x2=rd(),x3=rd(),x4=rd(),x5=rd(),x6=rd();
93         printf("%d\n",dijkstra(x1,x2,x3,x4,x5,x6));
94     }
95     return 0;
96 }

 

以上是关于luogu1979 华容道 (dijkstra+bfs)的主要内容,如果未能解决你的问题,请参考以下文章

洛谷P1979 华容道 [NOIP2013]

浴谷夏令营题单

P1979 华容道

luogu3645 [Apio2015]雅加达的摩天大楼 (分块+dijkstra)

luogu 4366 [Code+#4]最短路 Dijkstra + 位运算 + 思维

Luogu P4779dijkstra算法的堆优化