A*寻路之旅:用SDL图形化演示

news/2025/1/13 13:22:42/

前言

欢迎来到小K的数据结构专栏的第十小节,本节将为大家带来A*寻路算法的图形化详解,学了之后寻路不再迷路(✨当然也为大家准备了完整的源码,好像在文章顶部欸~ )~希望你看完之后,能对你有所帮助,不足请指正!共同学习交流 🐾

效果如下:

A*寻路算法图形化演示


文章目录

      • 前言
      • 一、简单介绍
      • 二、主要思想
      • 三、附上源码
      • 四、总结


一、简单介绍

由来
在 A * 算法之前有一种基于启发式探索的方法来提高Dijkstra算法的速度,这个算法叫做A1。后来的改进算法被称为A * 。 * 这个符号是从统计文献中借鉴来的,用来表示相对一个旧有标准的最优估计

启发式探索是利用问题拥有的启发信息来引导搜索,达到减少探索范围,降低问题复杂度的目的

✨A*寻路算法就是启发式探索的一个典型实践,在寻路的过程中,给每个节点绑定了一个估计值(即启发式),在对节点的遍历过程中是采取估计值优先原则,估计值更优的节点会被优先遍历。所以估计函数的定义十分重要,显著影响算法效率。
在这里插入图片描述

那么在上图中我们应该怎么评估出最短路径呐既然要评估,那肯定要有评估规则了,首先明确三个概念,H值,目前点到终点的曼哈顿距离(曼哈顿距离,就是两个位置长度差值和高度差值的和),G值,目前点到起点的消耗代价值,如果只是寻找路径,可以将该值也看成是这两点的曼哈顿距离,F值,H值和G值的和。所以A*寻路算法的评估由公示F=G+H来评估

✨我们先来尝试一下,假设每个格子的直线代价为10,斜线代价为14,则我们评估的起点周围的八个点的代价如下图所示:

在这里插入图片描述

怎么算的呐?我们以(2,2)为例,它到终点的曼哈顿距离我用黄色的矩形框起来了,横4纵4,然后乘上直线代价10,所以H为80,G一眼就可以看出,只有一个斜线代价,所以F为94

二、主要思想

在简单了解了A * 寻路算法了,我们不由得想,该怎么来寻?该用什么数据结构来描述?

  • ✨该怎么来寻?这个问题其实上边已经给出答案了,用F=G+H来评估,在这之前我们需要一个点类型,H比较好求,我们计算出当前点和终点之间的横纵坐标差,然后相加,乘上直线代价就好了,G值呐?我们通过下面的八叉树类型来解决~

    typedef struct Mypos 
    {int row, col;int f, g, h;
    }Mypos;
    //计算H值
    int getH(Mypos* pos, Mypos* endPos)
    {int x = ((pos->row > endPos->row) ? (pos->row - endPos->row) : (endPos->row - pos->row));int y = ((pos->col > endPos->col) ? (pos->col - endPos->col) : (endPos->col - pos->col));return (x + y) * ZXDJ;
    }
    
  • ✨该用什么数据结构?我首先想到的是八叉树,因为每个点周围都有八个点需要试探,下面是一个八叉树类型

    typedef struct MythreeNode 
    {Mypos pos;                                     //点struct MythreeNode* child[CHILD_NUM];          //孩子节点struct MythreeNode* partent;                   //父节点int child_Num;                                  //当前孩子数量
    }MythreeNode;
    
  • ✨具体的寻路过程如下

在这里插入图片描述

第一步,先遍历周围的八个节点,把他们的斜线代价计算出来

第二步,判断能不能走,能走就计算出F,存入树和数组中,不能走直接把该孩子删掉

第三步,从buffer数组中找到最小的F值,走,然后用辅助地图标记走过

第四步,我们要判断找没找到终点,退出循环有两种情况,要么是找到终点了,要么是buffer数组为空了

上述步骤中有一个小问题,就是如果遇到死胡同问题怎么办?比如下图:

在这里插入图片描述

第一步直接走到黄色的圈圈了,发现没路了,怎么办?我们思路回退一下,如果我们走完buffer数组中最小的,再把最小的删了不就可以了,这样下一步就会回到起点,这个问题就解决了

三、附上源码

✨A.h

#ifndef _A_H_
#define _A_H_
#include<stdio.h>
#include<stdlib.h>
#include<stdbool.h>
#include<string.h>
#include<assert.h>
#include<SDL.h>
//行列
#define ROWS 10
#define COLS 10
//代价
#define ZXDJ 10
#define XXDJ 14
//最大孩子数量
#define CHILD_NUM 8
//临时数组容量
#define NUMS_SIZE 1024
//路
enum type { road, wall };
//方向
enum Mydirect { p_up, p_down, p_left, p_right, p_upleft, p_upright, p_downleft, p_downright };
//点类型
typedef struct Mypos 
{int row, col;int f, g, h;
}Mypos;
//八叉树类型
typedef struct MythreeNode 
{Mypos pos;                                     //点struct MythreeNode* child[CHILD_NUM];          //孩子节点struct MythreeNode* partent;                   //父节点int child_Num;                                  //当前孩子数量
}MythreeNode;
//获得H值
int getH(Mypos* pos, Mypos* endPos);
//创建八叉树节点
MythreeNode* create_ThreeNode(Mypos* pos);
//判断能不能走
bool Can_Walk(Mypos* pos, bool map[ROWS][COLS], bool Pathmap[ROWS][COLS]);
//加载图片
SDL_Texture* load_BMP(SDL_Renderer* Ren, const char* fillname);
//绘图
void draw_Map(bool map[ROWS][COLS], Mypos* pos, SDL_Renderer* Ren, SDL_Texture** tex);#endif // _A_H_

✨A.c

#include "A.h"int getH(Mypos* pos, Mypos* endPos)
{int x = ((pos->row > endPos->row) ? (pos->row - endPos->row) : (endPos->row - pos->row));int y = ((pos->col > endPos->col) ? (pos->col - endPos->col) : (endPos->col - pos->col));return (x + y) * ZXDJ;
}MythreeNode* create_ThreeNode(Mypos* pos)
{MythreeNode* newNode = (MythreeNode*)malloc(sizeof(MythreeNode));if (NULL == newNode) return newNode;memset(newNode, 0, sizeof(MythreeNode));newNode->pos.row = pos->row;newNode->pos.col = pos->col;newNode->pos.g = pos->g;return newNode;
}bool Can_Walk(Mypos* pos, bool map[ROWS][COLS], bool Pathmap[ROWS][COLS])
{//越界if (pos->row < 0 || pos->row >= ROWS || pos->col < 0 || pos->col >= ROWS) return false;//是墙if (map[pos->row][pos->col]) return false;//走过if (Pathmap[pos->row][pos->col]) return false;return true;
}SDL_Texture* load_BMP(SDL_Renderer* Ren, const char* fillname)
{SDL_Surface* sfc = SDL_LoadBMP(fillname);if (!sfc){SDL_Log("sfc filed %s", SDL_GetError());return NULL;}SDL_Texture* tex = SDL_CreateTextureFromSurface(Ren, sfc);if (!tex){SDL_Log("tex failed %s", SDL_GetError());SDL_FreeSurface(sfc);return NULL;}SDL_FreeSurface(sfc);return tex;
}void draw_Map(bool map[ROWS][COLS], Mypos* pos,SDL_Renderer* Ren,SDL_Texture** tex)
{for (int i = 0; i < ROWS; i++){for (int j = 0; j < COLS; j++) {SDL_Rect rect = { j * 64,i * 64,64,64 };if (!map[i][j]){SDL_RenderCopy(Ren, tex[2], NULL, &rect);}else if (map[i][j]){SDL_RenderCopy(Ren, tex[3], NULL, &rect);}if (pos->row == i && pos->col == j) {SDL_RenderCopy(Ren, tex[0], NULL, &rect);}if (7 == i && 6 == j){SDL_RenderCopy(Ren, tex[1], NULL, &rect);}}}
}

✨main.c

在这里插入图片描述

四、总结

本节讲解的数据结构——A*寻路算法,他不仅是一种算法思想,它还是路径规划,游戏中普通人物挂机状态的寻路的灵魂,所以它是值得我们花费时间去掌握的~下节见!


http://www.ppmy.cn/news/122736.html

相关文章

使用网件无线路由器建立USB共享打印服务器

今天和大家分享一个netgare 无线wifi配置成USB打印机共享服务器。其实使用无线wifi实现usb打印机共享和airprint其实不算新鲜事&#xff0c;网上很多资料&#xff0c;但是今天的这个教程有亮点&#xff0c;不废话&#xff0c;上干货。 工具&#xff1a; 公司闲置的一无线路由器…

Vue--》Vue3打造可扩展的项目管理系统后台的完整指南(三)

今天开始使用 vue3 ts 搭建一个项目管理的后台&#xff0c;因为文章会将项目的每一个地方代码的书写都会讲解到&#xff0c;所以本项目会分成好几篇文章进行讲解&#xff0c;我会在最后一篇文章中会将项目代码开源到我的GithHub上&#xff0c;大家可以自行去进行下载运行&…

协众信息UI设计怎么提升用户体验呢

随着软件行业的发展兴起的一个新的设计行业。UI设计除了对美观有要求外&#xff0c;还对用户体验有要求&#xff0c;这也是UI设计不同于其他设计的地方。那么&#xff0c;UI设计怎么提升用户体验呢&#xff1f;   1、层次结构   要确保项目设计开始之前&#xff0c;就梳…

华为云认证有什么?考试难不难?

最近几年华为云的市场占比越来越大&#xff0c;逐渐占据了我们生活中的方方面面&#xff0c;而且很多政企单位&#xff0c;也选择华为云作为合作伙伴&#xff0c;因此市场上也需要越来越多的华为云人才&#xff0c;早在几年前&#xff0c;华为云就已经推出了自己的人才认证系统…

Language Models as Knowledge Embeddings:语言模型用作知识嵌入 IJCAI 2022

1.相关工作 1&#xff09;基于结构的知识嵌入 进一步分成基于翻译的模型和基于语义匹配的模型 基于翻译的模型采用基于距离的评分函数&#xff0c;TransE把实体和关系嵌入到一个维度为d的共享向量空间中&#xff1b;TransH,TransR,RotatE. 语义匹配模型采用基于相似性的评分函…

计算机开机需要注意什么,笔记本电脑第一次开机注意事项

语音内容&#xff1a; 大家好&#xff0c;我是时间财富网智能客服时间君&#xff0c;上述问题将由我为大家进行解答。 笔记本电脑第一次开机注意事项&#xff1a; 1、卸掉电池&#xff0c;主机的热量对电池寿命有影响。 2、不要用到一半就充电&#xff0c;虽然都是锂离子的&…

内胆包是保护你的MacBook的最佳选择

当你购买了一台昂贵的MacBook&#xff0c;你一定会想尽一切办法来保护它。但是&#xff0c;只有一个外壳可能不足以提供足够的保护。 在我使用MacBook的过程中&#xff0c;我发现了一个重要的配件&#xff1a;内胆包。内胆包是一种轻便的保护套&#xff0c;可以放入你的MacBoo…

来自ebay内部的「软件测试」学习资料,覆盖GUI、API自动化、代码级测试及性能测试等,Python等,拿走不谢!...

在软件测试领域从业蛮久了&#xff0c;常有人会问我&#xff1a; 刚入测试一年&#xff0c;很迷茫&#xff0c;觉得没啥好做的…… 测试在公司真的不受重视&#xff0c;我是不是去转型做开发会更好&#xff1f; 资深的测试架构师的发展路径是怎么样的&#xff1f;我平时该怎么…