算法入门经典上面的题。题目链接 uva816

大致题意

有一个最多包含9*9个交叉点的迷宫。输入起点、离开起点时的朝向和终点,求一条最短路(多解时任意输出一个即可)。详细题意请看原题

思路

其实就是bfs,但是难点在于转向,将四个方向和3种转弯方式编号为0~3和0~2。

下面是转弯函数的实现

const char *dirs="NESW";
const char *turns="FLR";
int dir_id(char c)
{
    return strchr(dirs,c)-dirs;
}

int turn_id(char c)
{
    return strchr(turns,c)-turns;
}

const int dr[]={-1, 0, 1,  0};
const int dc[]={ 0, 1, 0, -1};

node walk(const node &u,int turn)
{
    int dir=u.dir;
    if(turn==1) dir=(dir+3)%4;
    if(turn==2) dir=(dir+1)%4;
    return node(u.r+dr[dir],u.c+dc[dir],dir);
}

输入的时候读取交点的信息要注意处理。

利用has_edge[r][c][dir][turn]标记在坐标(r,c)处朝向dir转向turn能否成功。

AC代码

#include<iostream>//不需要判断边界、移动方向已经确定
#include<cstdio>
#include<cstring>
#include<queue>
#include<vector>
#include<sstream>
using namespace std;
int r0,c0,r1,c1,r2,c2;//r0、c0表示出发坐标 r2、c2是终点坐标

char dir0;//出发方向
const int maxn=11;
int has_edge[maxn][maxn][4][3],d[maxn][maxn][4];

struct node
{
    int r,c,dir;//r行 c列 dir方向
    node(int tr,int tc,int tdir)
    {
        r=tr;c=tc;dir=tdir;
    }
    node()
    {
    }
};
node p[maxn][maxn][4];

const char *dirs="NESW";
const char *turns="FLR";

int dir_id(char c)
{
    return strchr(dirs,c)-dirs;
}

int turn_id(char c)
{
    return strchr(turns,c)-turns;
}

const int dr[]={-1, 0, 1,  0};
const int dc[]={ 0, 1, 0, -1};

node walk(const node &u,int turn)
{
    int dir=u.dir;
    if(turn==1) dir=(dir+3)%4;
    if(turn==2) dir=(dir+1)%4;
    return node(u.r+dr[dir],u.c+dc[dir],dir);
}
void print_ans(node u)
{
    vector<node>nodes;
    for(;;)
    {
        nodes.push_back(u);
        if(d[u.r][u.c][u.dir]==0) break;
        u=p[u.r][u.c][u.dir];
    }

    //打印解
    int cnt=0;
    for(int i=nodes.size()-1;i>=0;i--)
    {
        if(cnt%10==0) printf(" ");
        printf(" (%d,%d)",nodes[i].r,nodes[i].c);
        if(++cnt%10==0) printf("\n");
    }
    if(nodes.size()%10!=0) printf("\n");
}
void bfs()
{
    int t=dir_id(dir0);
    d[r0][c0][t]=0;
    node temp(r0,c0,t);
    d[r1][c1][t]=1;
    node v(r1,c1,t);
    p[r1][c1][t]= temp;
    queue<node>q;
    q.push(v);

    while(!q.empty())
    {
    node u=q.front();
    q.pop();
    if(u.r==r2&&u.c==c2)
    {
        print_ans(u);
        return;
    }
    for(int i=0;i<3;i++)
    {
        if(has_edge[u.r][u.c][u.dir][i])
        {
            v=walk(u,i);
            if(d[v.r][v.c][v.dir]<0)
            {
                d[v.r][v.c][v.dir]=d[u.r][u.c][u.dir]+1;
                p[v.r][v.c][v.dir]=u;
                q.push(v);
            }
        }
    }
    }
    printf("  No Solution Possible\n");
}

int main(void)
{
    string name,pos;
    int r,c;

    string str;
    stringstream ss;
    while(cin>>name&&name!="END")
    {
        cin>>r0>>c0>>dir0>>r2>>c2;
        //计算进入迷宫的坐标
        if(dir0=='N')      {r1=r0-1;c1=c0;}
        else if(dir0=='E') {r1=r0;c1=c0+1;}
        else if(dir0=='W') {r1=r0;c1=c0-1;}
        else               {r1=r0+1;c1=c0;}

        memset(d,-1,sizeof(d));
        memset(has_edge,0,sizeof(has_edge));
        memset(p,0,sizeof(p)); 

        getchar();
        while(getline(cin,pos)&&pos!="0")
        {
            ss<<pos;
            ss>>r>>c;
            while(ss>>str)
            {
                if(str[0]!='*')
                {
                int dir=dir_id(str[0]);
                for(int i=1;i<str.size();i++)
                {
                int turn=turn_id(str[i]);
                has_edge[r][c][dir][turn]=1;
                }
                }
            }
            ss.clear();
        }
        cout<<name<<"\n";
        bfs();
    }
    return 0;
}

如有不当之处欢迎指出!

UVA 816 bfs的更多相关文章

  1. UVa 816 (BFS求最短路)

    /*816 - Abbott's Revenge ---代码完全参考刘汝佳算法入门经典 ---strchr() 用来查找某字符在字符串中首次出现的位置,其原型为:char * strchr (cons ...

  2. uva 816 BFS求最短路的经典问题……

    一开始情况没有考虑周全,直接WA掉了, 然后用fgets()出现了WA,给改成scanf就AC了 题目不是很难,用心就好…… #include <iostream> #include &l ...

  3. 【紫书】【重要】Abbott's Revenge UVA - 816 bfs 复杂模拟 带方向参数的迷宫

    题意:一个迷宫,每个交叉路口有一路标,限制了你从某方向进入该路口所能进入的路口. 题解:1.对于方向的处理:将node多增加一维dir,通过一个const 字符数组 加 上dir_id函数 以及一个方 ...

  4. UVA 816 - Abbott&#39;s Revenge(BFS)

    UVA 816 - Abbott's Revenge option=com_onlinejudge&Itemid=8&page=show_problem&category=59 ...

  5. UVA 816 -- Abbott's Revenge(BFS求最短路)

     UVA 816 -- Abbott's Revenge(BFS求最短路) 有一个 9 * 9 的交叉点的迷宫. 输入起点, 离开起点时的朝向和终点, 求最短路(多解时任意一个输出即可).进入一个交叉 ...

  6. uva 816 - Abbott&#39;s Revenge(有点困难bfs迷宫称号)

    是典型的bfs,但是,这个问题的目的在于读取条件的困难,而不是简单地推断,需要找到一种方法来读取条件.还需要想办法去推断每一点不能满足条件,继续往下走. #include<cstdio> ...

  7. UVa 816 Abbott的复仇(BFS)

    寒假的第一道题目,在放假回家颓废了两天后,今天终于开始刷题了.希望以后每天也能多刷几道题. 题意:这道BFS题还是有点复杂的,给一个最多9*9的迷宫,但是每个点都有不同的方向,每次进入该点的方向不同, ...

  8. BFS求最短路 Abbottt's Revenge UVa 816

    本题的题意是输入起点,朝向和终点,求一条最短路径(多解时任意输出一个即可) 本题的主要代码是bfs求解,就是以下代码中的slove的主要部分,通过起点按照路径的长度来寻找最短路径,输出最先到终点的一系 ...

  9. Abbott's Revenge UVA - 816 (输出bfs路径)

    题目链接:https://vjudge.net/problem/UVA-816 题目大意: 有一个最多包含9*9 个交叉点的迷宫.输入起点,离开起点时的朝向和终点,求一条最短路(多解时任意输出 一个即 ...

随机推荐

  1. cocos2dx 从2.2.6 到3.16 升级流水记录

    一个cocos2dx项目从2.2.6 升级至3.16 的过程,由于没有直升工具,类库升级也变动很大,有一部分需要手工完工升级.此记录供参考 1. 没有采用项目直升方式,先新建一个3.16的项目,然后把 ...

  2. MongoDB投影有$slice如何只显示该字段

    简单的投影 稍微用过MongoDB的都知道,投影很简单,就直接 db.student.find({_id:ObjectId('5a5085aed8f10c1a6cc0395b')},{comments ...

  3. SVN错误:Attempted to lock an already-locked dir的解决

    问题: SVN locked,文件被锁无法更新,SVN上更新代码失败,某些文件提示错误:Attempted to lock an already-locked dir 解决方法: 右键具体文件→Tea ...

  4. unity getcomponentsinchildren 翻船

    今天使用GetComponentsInChildren, 老司机翻船.因为一直以来我使用这个函数,下意识的从来所有的相同component都是放在子节点下,本身节点肯定不会放一个相同的componen ...

  5. POJ [P2631] Roads in the North

    树的直径 树的直径求法: 任取一点u,找到树上距u最远的点s 找到树上距s点最远的点t,s->t的距离即为所求 #include <iostream> #include <cs ...

  6. 济南清北学堂游记 Day 3.

    为什么你们都说下午题简单?我反而觉得下午题难...因为上午暴力很好写啊... 第一题惊了是一道博弈论,冷静推理一大波推出了正解,就变成了结论题. 可是结论题不都是水题吗? T2是一道关于异或的题,照例 ...

  7. BZOJ 2839: 集合计数 [容斥原理 组合]

    2839: 集合计数 题意:n个元素的集合,选出若干子集使得交集大小为k,求方案数 先选出k个\(\binom{n}{k}\),剩下选出一些集合交集为空集 考虑容斥 \[ 交集为\emptyset = ...

  8. Nodejs的运行原理-函数回调篇

    前言 当客户端向http server 发起TCP链接时,server端会发起一系列的callback调用,这是一个逆向调用的过程:开始于libuv,终止于js代码里的callback(promise ...

  9. zlib库VS2015编译步骤

    [点击这里下载zlib1.2.8源码](http://zlib.net/zlib128.zip) [点击这里下载zlib1.2.8编译动态库](http://zlib.net/zlib128-dll. ...

  10. 插上腾飞的翅膀:为asp.net core添加protobuf支持

    没时间解释了,快上车. 通过NuGet获取Zaabee.AspNetCoreProtobuf Install-Package Zaabee.AspNetCoreProtobuf 在Startup.cs ...