解救小哈 广搜 书中父亲f寻求路径

以下内容所提到的是啊哈算法原本就有的信息。转载请注明出处。

搜了很多网站,看过啊哈算法的人竟然没有想知道书中f是如何寻求路径的的,可能大佬们不看这种书吧,直接算法4或者算法导论起步。当然有可能啊哈算法确实比较小众,这些题都没人探讨探讨。

鉴于包括博客园搜索无果,我还是想写下我想了一下午的解决方案。看书难道不应该多想想吗?书中只给了f说题目不用求路径所以没用。那这个到底是怎么建立路径的?想清后对链表和指针理解又加深了一层,特别像数组模拟链表一样。

个人理解:结构体,链表,指针,都可以通过开数组解决。结构体是为了封装几个联系密切的变量。链表就像开两个数组,一个记录数据,另一个记录顺序。指针也就像多开一个数组,指向变量地址,就像另一个数组记录其对应位置。

啊哈算法给出f是这样进行的:

que[tail].f=head;

这么理解吧,结构体内f对应着它的上一个位置信息,即上一个指向que数组的第几个,所以从最后一个点开始,打印坐标然后按照f位置信息寻找对应的上一个坐标。

    int t;
    t=tail-1;
    while(t!=0)
    {
        printf("(%d,%d)\n",que[t].x,que[t].y);
        t=que[t].f;
    }
    printf("(startx,starty)");

因为最后一个0的条件就是起点 (startx,stary)了,所以结束不能忘了这个。完整代码如下:

#include <stdio.h>
#include <stdlib.h>
struct note
{
    int x;
    int y;
    int f;//父亲在队列中的编号(路径 指针)
    int s;//步数
};

int main()
{
    struct note que[2501];
	int head,tail;
	int i,j,t,k,n,m,startx,starty,p,q,tx,ty,flag;
    int a[51][51]={0},book[51][51]={0};
    int next[4][2]={{0,1},
                    {1,0},
                    {0,-1},
                    {-1,0}};
    scanf("%d %d",&n,&m);
    for(i=1;i<=n;i++)
        for(j=1;j<=m;j++)
            scanf("%d",&a[i][j]);
    scanf("%d %d %d %d",&startx,&starty,&p,&q);

    //队列初始化
    head=1;
    tail=1;
    //往队列里插入迷宫入口坐标
    que[tail].x=startx;
    que[tail].y=starty;
    que[tail].f=0;
    que[tail].s=0;
    tail++;
    book[startx][starty]=1;

    flag=0;

    while(head<tail)
    {
        for(k=0;k<=3;k++)
        {
            tx=que[head].x+next[k][0];
            ty=que[head].y+next[k][1];
            if(tx<1||tx>n||ty<1||ty>m)
                continue;
            if(a[tx][ty]==0&&book[tx][ty]==0)
            {
                //宽搜每个点只入队一次,所以和深搜不同,不需要book数组还原
                book[tx][ty]=1;
                que[tail].x=tx;
                que[tail].y=ty;
                que[tail].f=head;//这个点是从head扩展出来的,所以他的父亲是head
                que[tail].s=que[head].s+1;
                tail++;
            }
            if(tx==p&&ty==q)
            {
                flag=1;
                break;
            }
        }
        if(flag==1)
            break;
        head++;//当一个点扩展结束后,head++才能对后面的点在进行扩展
    }
    printf("%d\n",que[tail-1].s);
    //注意 链表和指针的理解 f相当于储存上一个表的地址
    t=tail-1;
    while(t!=0)
    {
        printf("(%d,%d)\n",que[t].x,que[t].y);
        t=que[t].f;
    }
    printf("(startx,starty)");

	system("pause");
	return 0;
}

  • 3
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值