Olya and Energy Drinks CodeForces - 877D(BFS+剪枝)

题目链接

题意:

一张网格图,有障碍物,每秒能向上下左右四个方向走1~k步,问从起点到终点的最短时间是多少,不能到达则输出-1.

 

分析:裸的BFS,但不剪枝会T。

 

剪枝1:

扩展时就判断是否到达终点

936ms/2000ms 物美价廉的剪枝

#include
using namespace std;
const int N=1005,INF=0x3f3f3f3f;
char s[N][N];
int xs,ys,xt,yt;
struct node{
    int x,y;
    int t;
};
//bool visx[N][N],visy[N][N];
bool vis[N][N];
int dir[][2]={{0,1},{0,-1},{1,0},{-1,0}};
int n,m,k;
bool bfs(){
    queueq;
    node v;
    v.x=xs,v.y=ys,v.t=0;
    q.push(v);
    //visx[xs][ys]=visy[xs][ys]=true;
    vis[xs][ys]=true;
    while(!q.empty()){
        auto e=q.front();
        q.pop();
        //cout<n || ny<=0 || ny>m || s[nx][ny]=='#' ) break;
                if(!vis[nx][ny]){ 
                    ne.x=nx,ne.y=ny,ne.t=e.t+1;
                    vis[nx][ny]=true;
                    q.push(ne);
                    //cout<<"push "<

 

剪枝2:

用d[nx][ny]数组记录从起点到(nx,ny)所用的步数,如果当前出队的点e的步数e.t已经大于等于d[nx][ny]就没必要再对这个点的这个方向扩展,直接跳出j的循环

62ms/2000ms 非常优秀的剪枝

#include
using namespace std;
const int N=1005,INF=0x3f3f3f3f;
char s[N][N];
int xs,ys,xt,yt;
struct node{
    int x,y;
    int t;
};
bool vis[N][N];
int d[N][N];//到起点的步数 更优才更新 剪枝
int dir[][2]={{0,1},{0,-1},{1,0},{-1,0}};

int n,m,k;
void bfs(){
    queueq;
    node v;
    v.x=xs,v.y=ys,v.t=0;
    q.push(v);
    vis[xs][ys]=true;
    d[xs][ys]=0;
    while(!q.empty()){
        auto e=q.front();
        q.pop();
        //cout<n||ny<1||ny>m||s[nx][ny]=='#'||d[nx][ny]<=e.t) break;
                if(!vis[nx][ny]){//呵呵
                    node ne;
                    ne.x=nx,ne.y=ny,ne.t=e.t+1;
                    vis[nx][ny]=true;
                    d[nx][ny]=ne.t;
                    q.push(ne);
                }
        
            }
        }
    }
    if(d[xt][yt]==INF)puts("-1");
    else printf("%d\n",d[xt][yt]);
}
int main(){
    
    scanf("%d%d%d",&n,&m,&k);
    for(int i=1;i<=n;++i){
        scanf("%s",s[i]+1);
    }
    for(int i=1;i<=n;++i){
        for(int j=1;j<=m;++j){
            d[i][j]=INF;
        }
    }
    scanf("%d%d%d%d",&xs,&ys,&xt,&yt);
    bfs();
    return 0;
}

PS:

break是直接跳出循环不再在这个方向继续往下搜,而不满足if只是不搜这一个点

不能盲目break,否则会漏解

你可能感兴趣的:(搜索/暴力,BFS,剪枝)