题目:迷宫2
有一个n*m的迷宫,迷宫第i行第j列上标着一个方向ci,j∈{L,R,U,D},牛妹站在格子(1,1),出口在格子(n,m)(n,m),牛妹想要走出迷宫,但牛妹只会按以下策略走:
设牛妹当前在格子(x,y)(x,y),则
- 若cx,y=L,则牛妹下一步走到格子(x,y-1)(x,y−1)
- 若cx,y=R,则牛妹下一步走到格子(x,y+1)(x,y+1)
- 若cx,y=U,则牛妹下一步走到格子(x-1,y)(x−1,y)
- 若c_{x,y}=D,则牛妹下一步走到格子(x+1,y)(x+1,y)
由于牛妹按这样的策略可能会无法走到出口,牛妹的好朋友牛牛决定在牛妹离开格子(1,1)前修改一些格子的方向,帮助牛妹走出迷宫,但是牛牛比较懒,他想要最小化修改的格子的数量,求牛牛至少需要修改多少个格子,此外牛牛还想知道修改的方案。
思路:
对每个格子,和自己的上下左右四个格子连边,当前方向指向的边边权为0,否则边权为1。
走过一条边权为1的边相当于进行了一次修改操作。
再跑一遍最短路就可以啦。
#include<bits/stdc++.h>
using namespace std;
namespace DIJKSTRA{
typedef long long ll;
typedef pair<int,ll> pil;
const int N=2e6;
const ll inf=0x3f3f3f3f3f3f3f3f;
vector<pil> g[N+5];
ll dist[N+5];
bool vis[N+5];
pil fa[N+5]; //记录路径
struct cmp{
bool operator () (pil e1,pil e2) {
return e1.second>e2.second;
}
};
priority_queue<pil,vector<pil>,cmp> que;
void init(int n,int st) {
for(int i=1;i<=n;i++) g[i].clear(),vis[i]=0,dist[i]=inf;
dist[st]=0;
}
void addedge(int u,int v,ll w) {
g[u].push_back({v,w});
}
ll dijkstra(int n,int st,int ed) {
que.push({st,0});
while(!que.empty()) {
int x=que.top().first;
que.pop();
if(vis[x]) continue;
else vis[x]=true;
for(pil t:g[x]) {
int y=t.first;
ll w=t.second;
if(dist[y]>dist[x]+w) {
dist[y]=dist[x]+w;
que.push({y,dist[y]});
fa[y]={x,w};
}
}
}
return dist[ed];
}
}
using namespace DIJKSTRA;
int n,m;
string a[N+5];
int getid(int x,int y) {
return (x-1)*m+y;
}
void slv() {
cin>>n>>m;
for(int i=1;i<=n;i++) {
string s;
cin>>s;
a[i]=" "+s;
}
init(n*m,1);
const int mv1[5]={0,-1,1,0,0};
const int mv2[5]={0,0,0,-1,1};
for(int i=1;i<=n;i++) {
for(int j=1;j<=m;j++) {
int t[5]={0,1,1,1,1};
if(a[i][j]=='U') t[1]=0;
else if(a[i][j]=='D') t[2]=0;
else if(a[i][j]=='L') t[3]=0;
else t[4]=0;
for(int k=1;k<=4;k++) {
int x=i+mv1[k],y=j+mv2[k];
if(x>0&&y>0&&x<=n&&y<=m) addedge(getid(i,j),getid(x,y),t[k]);
}
}
}
cout<<dijkstra(n*m,1,n*m)<<endl;
int t=n*m;
while(t!=1) {
int u=fa[t].first;
if(fa[t].second!=0) {
int x=(u-1)/m+1,y=((u%m==0)?m:u%m);
cout<<x<<' '<<y<<' ';
if(u-m==t) cout<<"U\n";
else if(u+m==t) cout<<"D\n";
else if(u-1==t) cout<<"L\n";
else cout<<"R\n";
}
t=u;
}
return ;
}
int main() {
std::ios::sync_with_stdio(false);
int T;
cin>>T;
while(T--) {
slv();
}
return 0;
}