POJ 1573 Robot Motion(模拟、DFS、BFS)

题目链接:http://poj.org/problem?id=1573

分析:

这道题既可以模拟出结果,也可以用DFS或者BFS搜索。

模拟:

// 模拟
// #include <bits/stdc++.h>
#include <cstdio>
#include <cstring>
using namespace std;

char G[10+7][10+7];
int vis[10+7][10+7];

int main()
{
    int n, m, s;
    while(scanf("%d %d %d", &n, &m, &s), n+m+s)
    {
        memset(vis, 0, sizeof(vis));
        for(int i=0; i<n; i++)
            scanf("%s", G[i]);
        // for(int i=0; i<n; i++)
        //     printf("%s\n", G[i]);
        int sx = 0, sy = s-1;
        int step1 = 0, step2 = 0;
        int flag = 0;
        while(1)
        {
            if(G[sx][sy] == 'N')
                vis[sx][sy] = 1, sx--;
            else if(G[sx][sy] == 'E')
                vis[sx][sy] = 1, sy++;
            else if(G[sx][sy] == 'S')
                vis[sx][sy] = 1, sx++;
            else if(G[sx][sy] == 'W')
                vis[sx][sy] = 1, sy--;
            step1 ++;   // 记录走了几步
            
            if(sx < 0 || sy < 0 || sx >= n || sy >= m)  // 判断是否走出方格
            {
                flag = 1;
                break;
            }
            else if(vis[sx][sy] == 1)   // 判断是否陷入循环,并标记
            {
                vis[sx][sy] = 2;
                flag = 2;
                break;
            }
        }
        if(flag == 2)   // 如果进入了循环
        { // 把循环部分再走一遍并记录循环过程中走了几步,其实可以用个二维数组记录一下每个点是第几步走到的,这样就可以不用再走一遍循环部分
            while(1)    
            {
                 if(G[sx][sy] == 'N')
                    sx--;
                else if(G[sx][sy] == 'E')
                    sy++;
                else if(G[sx][sy] == 'S')
                    sx++;
                else if(G[sx][sy] == 'W')
                    sy--;
                
                step2 ++;
                if(vis[sx][sy] == 2)    // 如过走到了循环起点,就结束
                    break;
            } 
        }
        if(flag == 1)
        {
            // if(step1 > 1)
                printf("%d step(s) to exit\n", step1);
            // else
                // printf("%d step to exit\n", step1);
        }
        else if(flag == 2)
        {
            // if(step1 > 1)
                printf("%d step(s) before a loop of %d step(s)\n", step1 - step2, step2);
            // else
                // printf("%d step before a loop of %d step(s)\n", step1 - step2, step2);
        }
    }




    return 0;
}

 

DFS:

// 递归写法
// DFS
#include <cstdio>
#include <cstring>
using namespace std;

char G[20][20];
int step[20][20];
int n, m, s;
int exitStep;
int loopStep;
int sx, sy;
bool loop = false;

void dfs(int row, int col, int num)
{
    if(row < 0 || col < 0 || row >= n || col >= m)  // 走出方格条件
    {
        exitStep = num;
        return ;
    }

    if(step[row][col])  // 有走回到了原来走过的点,说明是一个循环
    {
        loop = true;
        loopStep = num - step[row][col];    // 循环步数
        sx = row;
        sy = col;
        return;
    }

    step[row][col] = num;   // 起点不能标记为0,因为我们以非零标记此点是否走过
    if(G[row][col] == 'N')
        dfs(row-1, col, num+1);
    else if(G[row][col] == 'E')
        dfs(row, col+1, num+1);
    else if(G[row][col] == 'S')
        dfs(row+1, col, num+1);
    else if(G[row][col] == 'W')
        dfs(row, col-1, num+1);
}

int main()
{
   
    while(scanf("%d %d %d", &n, &m, &s), n+m+s)
    {
        memset(step, 0, sizeof(step));
        for(int i=0; i<n; i++)
        {
            scanf("%s", G[i]);
        }

        loop = false;
        dfs(0, s-1, 1);
        // for(int i=0; i<n; i++)
        // {
        //     for(int j=0; j<m; j++)
        //         printf("%d", step[i][j]);
        //     printf("\n");
        // }
        if(loop)
            printf("%d step(s) before a loop of %d step(s)\n", step[sx][sy] - 1, loopStep);
        else
        {
            printf("%d step(s) to exit\n", exitStep -1);
        }
        
    }


    return 0;
}

 

BFS:

// BFS
#include <cstdio>
#include <cstring>
#include <queue>
using namespace std;

struct node{
    int x, y;
    node(int xx=0, int yy=0) : x(xx), y(yy) {}  // 构造函数
};

char G[20][20];
int step[20][20];
int n, m , s;
int numstep;    // 记录走了多少步
bool loop = false;  // 记录是否有循环
int loopstep;
int loopx, loopy;   // 循环起点

void bfs()
{
    queue<node> q;
    q.push(node(0, s-1));   // 起点放入队列中
    while(!q.empty())
    {
        node e = q.front();
        q.pop();
        

        if(step[e.x][e.y])  // 如果已经走过了, 说明陷入循环
        {
            loop = true;
            loopx = e.x;
            loopy = e.y;
            loopstep = numstep - step[e.x][e.y];
            break;
        }

        numstep ++;
        step[e.x][e.y] = numstep;   // 标记当前点已经走过,而且是第几步走到这里的

        if(G[e.x][e.y] == 'N')
            e.x -= 1;
        else if(G[e.x][e.y] == 'E')
            e.y += 1;
        else if(G[e.x][e.y] == 'S')
            e.x += 1;
        else if(G[e.x][e.y] == 'W')
            e.y -= 1;
        
        if(e.x >=0 && e.y >= 0 && e.x < n && e.y < m)
                q.push(e);
        else            // 出了方格就退出
            break;
        
    }
}

int main()
{
    while(scanf("%d %d %d", &n, &m, &s), n+m+s)
    {
        for(int i=0; i<n; i++)
            scanf("%s", G[i]);
        memset(step, 0, sizeof(step));
        loop = false;
        numstep = 0;
        bfs();

        if(loop)
            printf("%d step(s) before a loop of %d step(s)\n", step[loopx][loopy]-1, loopstep+1);
        else
        {
            printf("%d step(s) to exit\n", numstep);
        }
    }


    return 0;
}

 

全部评论

相关推荐

其实本来打算等lastday的时候再写的,但是现在提笔写下这篇总结完全是出于自己的想法,今天上午自己被学校发的签到吵醒时才突然想明白了很多事情,遂决定写下本文进行总结,虽然现在顶多算2.5个月,但也大差不差喵。回看这段时间的日常实习,我的关键词是:遗憾,焦虑。当然也有快乐的时候,不过大部分时间都是前面这两种情绪主导。为了避免后人再次踩坑,我将在本文详细解释我遇到的困难&nbsp;+&nbsp;产生的原因&nbsp;+&nbsp;应对的措施。同时总结新人实习时除了业务本身,还有如何处理生活与工作上的平衡,调控自身的情绪,让自己恢复到最好的工作状态。本文不会教你实习怎么去做产出,因为有产出的前提是你的心态足够健康,且在工作之余还有时间去...
wuwuwuoow:你的经历跟挺像,但我实力远没你强,现在只能干外包。但解决焦虑这块我应该比你更有经验,因为我曾经也非常迷茫和焦虑: 1.规律作息。无论节假日,都必须在同一时间点睡觉,同一时间点起床。放假睡的多,工作睡的少,这就是典型的作息不规律。将直接干扰前额叶皮层功能,导致情绪波动(易怒、焦虑)。无论上班还是周末,我都是 11:30 睡,7 点起床。7.5h 睡眠,完全足够了。 2.运动。缓解压力,强身健体,提高免疫力。不要觉得每天没有时间锻炼,都是懒惰的借口。 3.冥想。长期练习会增厚前额叶皮层(理性决策区),缩小杏仁核体积(减少情绪过敏反应,核心),增强情绪调控能力。 方法很简单,任何时候都能做。就是闭上眼睛,只专注自己的呼吸,不去想其他任何事情。你可以尝试一下,你会发现非常难只专注呼吸,会有大量的想法涌现出来(什么走马灯),不要去压抑它们,而是放平心态,把注意力继续放在呼吸上面。 而且最重要的是,冥想让你学会“活在当下”。因为处于冥想的你,除了专注呼吸你还能做什么呢?你什么都做不了。生活也是这样,我们无法改变过去,无法预知未来会发生什么,我们能做的只有手头的事情,除此之外什么都别想,因为你无法去改变它们。 4.工作与生活分离。工作不是生活的全部,生活可不是只有工作。像我放假的时候,从不带电脑回去。放假该玩就玩吧。 上面要是都能做到,其实完全解决不了你工作上的问题,完不成的需求还是完不成,面试该挂还是得挂。不过呢,当你再次迷茫,再次焦虑的时候,你会发现,诶,还好,没这么难受。比如面试挂了,可能以前的你会感觉非常难受。但如果你做到以上 4 点,你还是会难受的,但其实又没这么难受,可能你会这样想:既然挂了我还能怎么样?这公司不要我,有的是公司要我!
投递腾讯等公司6个岗位 >
点赞 评论 收藏
分享
purcoter:虚拟货币预测正确率百分之99,还要找工作干嘛,不早就财富自由了
点赞 评论 收藏
分享
评论
点赞
收藏
分享

创作者周榜

更多
牛客网
牛客企业服务