zoukankan      html  css  js  c++  java
  • CodeForces CF877D题解(BFS+STL-set)

    解法(1:)

    正常的(bfs)剪枝是(Theta(nm4k)),这个时间复杂度是只加一个(vis)记录的剪枝的,只能保证每个点只进队一次,并不能做到其他的减少时间,所以理论上是过不了的。(n*m*4*k=4*10^9)

    这里提供一个(codeforces)的官方解法:(STL-set)优化(bfs)。使用(set)主要是能够快速找到当前行和列有那个点没有被找到过,每行每列单独建立一个(set)(set<int> line[N],col[N])

    (bfs)的时候(set<int>::iterator it=line[x].lower\_bound(y))找两次,分别处理左边和右边的答案。

    (set<int>::iterator it=col[y].lower\_bound(x)),分别处理上下方向的答案。

    细节在代码里。

    #include<bits/stdc++.h>
    using namespace std;
    const int N=1010;
    set<int> lin[N],col[N];
    int n,m,k;
    char mp[N][N];
    int sx,sy,ex,ey;
    queue<pair<int,pair<int,int> > > q;
    int read()
    {
        int x=0,f=1;
        char c=getchar();
        while (c<'0'||c>'9')
        {
            if(c=='-') f=-1;
            c=getchar();
        }
        while (c>='0'&&c<='9')
        {
            x=(x<<1)+(x<<3)+(c^48);
            c=getchar();
        }
        return x*f;
    }
    void add_to_que(int x,int y,int dis)
    {
        q.push(make_pair(dis,make_pair(x,y)));
        lin[x].erase(y);
        col[y].erase(x);
    }
    void BFS()
    {
        //int x1=read(),y=read()
        add_to_que(sx,sy,0);
        while (!q.empty())
        {
            int curx=q.front().second.first;
            int cury=q.front().second.second;
            int curd=q.front().first;
            q.pop();
    
            if(curx==ex && cury==ey)
            {
                printf("%d
    ",curd);
                exit(0);
            }
            set<int>::iterator wsy=lin[curx].lower_bound(cury);
    
            while (wsy!=lin[curx].end() && *wsy-cury<=k && mp[curx][*wsy]!='#')
            {
                add_to_que(curx,*wsy,curd+1);
                wsy=lin[curx].lower_bound(cury);
            }
            wsy=lin[curx].lower_bound(cury);
            while (wsy!=lin[curx].begin() && cury-*(--wsy)<=k && mp[curx][*wsy]!='#')
            {
                add_to_que(curx,*wsy,curd+1);
                wsy=lin[curx].lower_bound(cury);
            }
    
            wsy=col[cury].lower_bound(curx);
            while (wsy!=col[cury].end() && *wsy-curx<=k && mp[*wsy][cury]!='#')
            {
                add_to_que(*wsy,cury,curd+1);
                wsy=col[cury].lower_bound(curx);
            }
            wsy=col[cury].lower_bound(curx);
            while (wsy!=col[cury].begin() && curx-*(--wsy)<=k && mp[*wsy][cury]!='#')
            {
                add_to_que(*wsy,cury,curd+1);
                wsy=col[cury].lower_bound(curx);
            }
        }
        puts("-1");
    }
    int main()
    {
        n=read(),m=read(),k=read();
        for(int i=1;i<=n;i++) for(int j=1;j<=m;j++) cin>>mp[i][j],lin[i].insert(j),col[j].insert(i);
        sx=read(),sy=read(),ex=read(),ey=read();
        BFS();
        return 0;
    }
    

    解法(2:)

    卢神发现的一个(Theta(nm))的算法(()我也不知道是不是这个复杂度())

    每次找一个点的时候,如果在更新答案时发现当前这个新的答案不比之前的更优的话,就直接(break),是的,直接退出循环。

    为什么这样是对的呢?如果答案是从同一条路上来的,那么显然将来的答案不会更优。如果是拐弯过来的,那么显然按照之前的走更优。

    #include<bits/stdc++.h>
    #define int long long
    using namespace std;
    int read()
    {
        int x=0,f=1;
        char c=getchar();
        while (c<'0'||c>'9')
        {
            if(c=='-') f=-1;
            c=getchar();
        }
        while (c>='0'&&c<='9')
        {
            x=(x<<1)+(x<<3)+(c^48);
            c=getchar();
        }
        return x*f;
    }
    const int N=1005,inf=0x3f3f3f3f3f3f3f3f;
    int n,m,k,sx,sy,ex,ey,dis[N][N],ans;
    int dx[]={0,0,-1,1};
    int dy[]={-1,1,0,0};
    char ch[N][N];
    queue<pair<int,int> >q;
    int dij()
    {
        memset(dis,0x3f,sizeof(dis));
        q.push(make_pair(sx,sy));
        dis[sx][sy]=0;
        while(!q.empty())
        {
            int x=q.front().first,y=q.front().second; q.pop();
            for(int i=0;i<4;i++)
            {
                for(int j=1;j<=k;j++)
                {
                    int nx=x+dx[i]*j,ny=y+dy[i]*j;
                    if(nx<1 || nx>n || ny<1 || ny>m || ch[nx][ny]=='#' || dis[nx][ny]<dis[x][y]+1) break;
                    if(dis[nx][ny]==inf)
                    {
                        dis[nx][ny]=dis[x][y]+1;
                        q.push(make_pair(nx,ny));
                    }
                }
            }
        }
        return dis[ex][ey];
    }
    signed main()
    {
        n=read(),m=read(),k=read();
        for(int i=1;i<=n;i++) scanf("%s",ch[i]+1);
        sx=read(),sy=read(),ex=read(),ey=read();
        if(sx==ex && sy==ey) puts("0"),exit(0);
        ans=dij();
        printf("%lld",ans==inf?-1:ans);
        return 0;
    }
    
    
  • 相关阅读:
    贾鹏芳 二胡演奏家
    php 关于下载中文文件实现
    Android 线程。。
    Hadoop命令行接口运行自己编写的类
    ubuntu下jdk环境变量的设置
    navicat导入sql语句文件失败原因-----datetime
    sql server获取当前时间到秒级和毫秒级的语句
    sql关于group by的问题
    java的equal和==的区别
    win8.1 占用80端口问题
  • 原文地址:https://www.cnblogs.com/juruo-wsy/p/14622370.html
Copyright © 2011-2022 走看看