2022牛客寒假算法基础集训营6 G迷宫2

题目:迷宫2

有一个n*m的迷宫,迷宫第i行第j列上标着一个方向ci,j∈{L,R,U,D},牛妹站在格子(1,1),出口在格子(n,m)(n,m),牛妹想要走出迷宫,但牛妹只会按以下策略走:
设牛妹当前在格子(x,y)(x,y),则

  1. 若cx,y=L,则牛妹下一步走到格子(x,y-1)(x,y−1)
  2. 若cx,y=R,则牛妹下一步走到格子(x,y+1)(x,y+1)
  3. 若cx,y=U,则牛妹下一步走到格子(x-1,y)(x−1,y)
  4. 若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;
} 
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值