hdoj 1728 逃离迷宫
题目:
问你能不能在k次转弯内,由起点到达终点。换言之,求起点到终点的最小转弯数。
感觉这类问题也是特别经典的,有点像由起点到终点的最小步数。
那么,什么是转弯?例如样例1:
我们可以这样理解就像画一条直线一样,从原点向4个方向延伸画四条直线(蓝色),遇到*停止(因为你不能穿墙啊),在从这4条直线路过的点重复上
面的操作,注意如果有交点以前面那个为准,因为你每画一条线相当于转一次弯,当然是以前面那个为准。
对于这个题的实现可以用visit数组来存储到某一个点转了几次弯,也可以用结构体来封装。没什么本质上的区别,可以相当于拿来练练手
code:
/*用visit数组记录转了几次弯*/
#include<cstdio> #include<cstring> #include<queue> using namespace std; struct node{int x,y;}s; const int MAXN=100+5; char a[MAXN][MAXN]; int visit[MAXN][MAXN]; int k,sy,sx,ey,ex,m,n; int dx[4]={-1,1,0,0},dy[4]={0,0,1,-1}; bool judge(int x,int y){ return x>=1&&x<=m&&y>=1&&y<=n&&a[x][y]!='*'; } void bfs(){ memset(visit,-1,sizeof(visit)); queue<node>q;while(!q.empty())q.pop(); s.x=sx;s.y=sy; q.push(s); while(!q.empty()){ node temp=q.front();q.pop(); for(int i=0;i<4;++i){ node no;//以temp.x,temp.y为原点向四个方向运动 no.x=temp.x+dx[i]; no.y=temp.y+dy[i]; while(judge(no.x,no.y)){ if(visit[no.x][no.y]==-1){//如果下一个点还没有访问过,do,否则,跳过更新这个点。 visit[no.x][no.y]=visit[temp.x][temp.y]+1; q.push(no);//把路过的点加入队列 if(no.x==ex&&no.y==ey){//到达了终点 printf(visit[no.x][no.y]<=k?"yes\n":"no\n"); return; } } no.x+=dx[i];no.y+=dy[i];//沿着方向,不忘初心,继续前行 } } } printf("no\n");//无法到达终点 } int main(){ int T;scanf("%d",&T); while(T--){ scanf("%d%d",&m,&n);getchar(); for(int i=1;i<=m;++i){ for(int j=1;j<=n;++j) scanf("%c",&a[i][j]); getchar(); } scanf("%d%d%d%d%d",&k,&sy,&sx,&ey,&ex); if(sx==ex&&sy==ey)printf("yes\n"); //这儿要加特判,具体的,可以把下面打出来看看,是因为初始化的原因。 else bfs(); /*for(int i=1;i<=m;++i){ for(int j=1;j<=n;++j) printf("%d\t",visit[i][j]); printf("\n"); }*/ } }
code:
/*用visit数组标记是否被访问过*/
#include<cstdio> #include<cstring> #include<queue> using namespace std; struct node{int x,y,turn;}s; const int MAXN=100+5; char a[MAXN][MAXN]; int visit[MAXN][MAXN]; int sx,sy,ex,ey,k,m,n; int dx[4]={-1,1,0,0,},dy[4]={0,0,1,-1}; bool judge(int x,int y){ return x>=1&&x<=m&&y>=1&&y<=n&&a[x][y]!='*'; } void bfs(){ memset(visit,0,sizeof(visit)); queue<node>q;while(!q.empty())q.pop(); s.x=sx;s.y=sy;s.turn=-1; q.push(s); while(!q.empty()){ node no=q.front();q.pop(); for(int i=0;i<4;++i){ node temp; temp.x=no.x+dx[i]; temp.y=no.y+dy[i]; while(judge(temp.x,temp.y)){ if(!visit[temp.x][temp.y]){ visit[temp.x][temp.y]=1; temp.turn=no.turn+1; q.push(temp); if(temp.x==ex&&temp.y==ey){ printf(temp.turn<=k?"yes\n":"no\n"); return ; } } temp.x+=dx[i];temp.y+=dy[i]; } } } printf("no\n"); } int main(){ int T;scanf("%d",&T); while(T--){ scanf("%d%d",&m,&n);getchar(); for(int i=1;i<=m;++i){ for(int j=1;j<=n;++j) scanf("%c",&a[i][j]); getchar(); } scanf("%d%d%d%d%d",&k,&sy,&sx,&ey,&ex); if(sx==ex&&sy==ey)printf("yes\n"); else bfs(); /*for(int i=1;i<=m;++i){ for(int j=1;j<=n;++j) printf("%d\t",visit[i][j]); printf("\n"); }*/ } }
未完待续。