A*启发式搜索基础

传统的搜索方式是盲目搜索,即到每一步的时候并没有对每种情况进行有效的区分,这样的结果是浪费了大量的时间,对很多没有必要的数据进行了搜索。
而A*算法则在搜索的过程中会选取认为“最优”的状态进行搜索,而这正是这种算法的精华部分。

其实我们可以将他和Dijkstra进行一定的对比,他们的共同点很多,都是每次选取最优的点开始搜索,但是他们的“最优”策略不同也就决定了不同的效率。

  • Dijkstra是每次选取距离“出发点”最近的点开始搜索
  • A*算法则是选取距离“目标点”估计值最小的点开始搜索

注意这里的估计值就是A*算法的核心部分了,本博文最后会介绍和Dijkstra的具体效率的区别。

A*算法将点到重点的距离分为两个部分:

F(n)=G(n)+H(n)

介绍:

  • F(n)表示起点到目标的估计距离
  • G(n)表示起点到n点的准确距离
  • H(n)表示n点到目标点的估计距离

我们先假设从n点到目标点的准确距离为H′(n),当然H′(n)的准确值我们并不知道,否则我们就不用费力气去求了。
这里H(n)需要满足的条件是

  • H(n)<=H′(n)

需要说明的是,我们A*搜索的核心就是H(n)的构建,最后构建的好坏直接决定了我们搜索的效率,我们的好是指:H(n)越贴近H′(n)越好,但是注意一定不能超过实际H′(n)的值。

为了和大多数的讲解名字对应,本博文采用相同名字,即名openlist的优先队列来存放需要搜索的内容,具体搜索步骤如下

  • 预定义H(n)=|beginx−endx|+|beginy−endy|
  • 预定义F(n)=G(n)+H(n)
  • openlist中F(n)值小的优先级别高

    1. 将起点加入openlist中
    2. 从队列中弹出元素
    3. 对弹出元素的周边进行搜索,如果发现有未被搜索到的位置,并且不是墙,河水,峡谷等不能行走的位置,则将其加入到队列中,并将其指向弹出元素。
      如果发现有已经被搜索过的位置,则查看其F(n)值是否比现在走到那个位置的F(n)值小,若现在是更优解,则更新其F(n)值。如果其不在队列中则把其重新加入队列,否则什么也不用做。
    4. 重复2-4步,直到找到目标点
    5. 最小路径即目标点的F(n)的值,同时根据维护的指针就可把最优路径推出来

补充:

我们从F(n)=G(n)+H(n)中可以推导出

  • 当G(n)=0时,算法相当与bfs;
  • 当H(n)=0时,算法相当于dfs;

自己写了一个示例代码如下:

@Frosero
#include <iostream>
#include <queue>
#include <cstdlib>
#include <cstring>
#include <cstdio>
#define INF 0x3f3f3f3f
using namespace std; char mps[12][21] = { //x*
{"...................."}, //0
{"...S................"}, //1
{"...................."}, //2
{"...................."}, //3
{"....##########......"}, //4
{"...................."}, //5
{"...................."}, //6
{"...................."}, //7
{".............#......"}, //8
{".............#..P..."}, //9
{".............#......"}, //10
{"...................."}, //11
}; //原地图 const int beginx = 1,beginy = 3,endx = 9,endy = 16;
int link[12][21] = {0}; //1-up 2-right 3-down 4-left
int dx[] = {1,-1,0,0},dy[] = {0,0,1,-1},f[12][21]; //f用于储存F值 bool can_move(int x,int y){
return x >= 0 && x < 12 && y >= 0 && y < 20 && mps[x][y] != '#';
} class Node{
public:
int x,y;
int f,g,h;
Node() = default;
Node(int X,int Y,int G):x(X),y(Y),g(G) {
this->h = abs(x - endx) + abs(y - endy) ; //计算H(n)
this->f = this->g + this->h;
}
bool operator < (const Node &rhs) const {
return this->f > rhs.f;
}
}; void A_bfs(){
for(int i=0;i<12;i++) for(int j=0;j<21;j++) f[i][j] = INF;
priority_queue<Node> openlist;
while(!openlist.empty()) openlist.pop();
openlist.push(Node(beginx,beginy,0));
f[beginx][beginy] = abs(beginx - endx) + abs(beginy - endy) ;
while(!openlist.empty()){
Node now = openlist.top(); openlist.pop();
if(now.x == endx && now.y == endy) break;
if(now.f > f[now.x][now.y]) continue; //如果发现刚才记录的F值比现在实际F值小
//说明该点已经被更新过,忽略即可
for(int i=0;i<4;i++){
int nx = now.x + dx[i];
int ny = now.y + dy[i];
if(can_move(nx,ny) && now.g + 1 + abs(nx - endx) + abs(ny - endy) < f[nx][ny]){
mps[nx][ny] = mps[nx][ny] == 'P' ? 'P' : '*' ;
f[nx][ny] = now.g + 1 + abs(nx - endx) + abs(ny - endy) ;
if(nx == now.x + 1) link[nx][ny] = 1;
else if(nx == now.x - 1) link[nx][ny] = 3;
else if(ny == now.y + 1) link[nx][ny] = 4;
else if(ny == now.y - 1) link[nx][ny] = 2; //标记指针数组
openlist.push(Node(nx,ny,now.g + 1));
}
}
}
} int main(){
A_bfs();
int x = endx,y = endy;
while(1){
if(link[x][y] == 1) x--;
else if(link[x][y] == 2) y++;
else if(link[x][y] == 3) x++;
else if(link[x][y] == 4) y--;
if(x == beginx && y == beginy) break;
mps[x][y] = '+';
} printf("ans is : %d\n",f[endx][endy]);
for(int i=0;i<12;i++,printf("\n")){
for(int j=0;j<20;j++) printf("%c",mps[i][j]);
} return 0;
} /* 输出: ans is : 21
...*****............
..*S************....
..*+*************...
..*+************....
..*+##########***...
..*+++++++++++++**..
..*************+**..
...************+**..
....*********#*+**..
.....********#*+P...
......*.*...*#.*....
.................... 符号解释
S:起点
P:目标点
*:搜索过的路径
+:搜索过的最优路径
#:墙 Process returned 0 (0x0) execution time : 0.362 s
Press any key to continue. */

作为比较,我们用Dijkstra再做一次结果:

@Frosero
#include <iostream>
#include <queue>
#include <cstdlib>
#include <cstring>
#include <cstdio>
#define INF 0x3f3f3f3f
using namespace std; char mps[12][21] = { //x*
{"...................."}, //0
{"...S................"}, //1
{"...................."}, //2
{"...................."}, //3
{"....##########......"}, //4
{"...................."}, //5
{"...................."}, //6
{"...................."}, //7
{".............#......"}, //8
{".............#..P..."}, //9
{".............#......"}, //10
{"...................."}, //11
}; const int beginx = 1,beginy = 3,endx = 9,endy = 16;
int link[12][21] = {0}; //1-up 2-right 3-down 4-left
int dx[] = {1,-1,0,0},dy[] = {0,0,1,-1},f[12][21]; bool can_move(int x,int y){
return x >= 0 && x < 12 && y >= 0 && y < 20 && mps[x][y] != '#';
} class Node{
public:
int x,y;
int g;
Node() = default;
Node(int X,int Y,int G):x(X),y(Y),g(G) { }
bool operator < (const Node &rhs) const {
return this->g > rhs.g;
}
}; void A_bfs(){
for(int i=0;i<12;i++) for(int j=0;j<21;j++) f[i][j] = INF;
priority_queue<Node> openlist;
while(!openlist.empty()) openlist.pop();
openlist.push(Node(beginx,beginy,0));
f[beginx][beginy] = 0 ;
while(!openlist.empty()){
Node now = openlist.top(); openlist.pop();
if(now.x == endx && now.y == endy) break;
for(int i=0;i<4;i++){
int nx = now.x + dx[i];
int ny = now.y + dy[i];
if(can_move(nx,ny) && now.g + 1 < f[nx][ny]){
mps[nx][ny] = mps[nx][ny] == 'P' ? 'P' : '*' ;
f[nx][ny] = now.g + 1;
if(nx == now.x + 1) link[nx][ny] = 1;
else if(nx == now.x - 1) link[nx][ny] = 3;
else if(ny == now.y + 1) link[nx][ny] = 4;
else if(ny == now.y - 1) link[nx][ny] = 2;
openlist.push(Node(nx,ny,now.g + 1));
}
}
}
} int main(){
A_bfs();
int x = endx,y = endy;
while(1){
if(link[x][y] == 1) x--;
else if(link[x][y] == 2) y++;
else if(link[x][y] == 3) x++;
else if(link[x][y] == 4) y--;
if(x == beginx && y == beginy) break;
mps[x][y] = '+';
} printf("ans is : %d\n",f[endx][endy]);
for(int i=0;i<12;i++,printf("\n")){
for(int j=0;j<20;j++) printf("%c",mps[i][j]);
} return 0;
} /* 输出: ans is : 21
********************
***S++++++++++******
*************+******
*************++*****
****##########+*****
**************+*****
**************+*****
**************+*****
*************#+****.
*************#++P...
*************#**....
***************..... 符号解释
S:起点
P:目标点
*:搜索过的路径
+:搜索过的最优路径
#:墙 Process returned 0 (0x0) execution time : 0.362 s
Press any key to continue. */

通过比较我们可以很轻松的发现A*搜索相比于Dijkstra的优越性

下面是在网上找的几组图片来进行一些对比
图片来源:[http://blog.csdn.net/luckyxiaoqiang/article/details/6996963]

例一

1. A*(曼哈顿距离)

2. A*(欧氏距离)

3. A*(切比雪夫距离)

4. Dijkstra

5. 双向BFS

例二

1. A*(曼哈顿距离)

2. A*(欧氏距离)

3. A*(切比雪夫距离)

4. Dijkstra

5. 双向BFS

例三

1. A*(曼哈顿距离)

2. A*(欧氏距离)

3. A*(切比雪夫距离)

4. Dijkstra

5. 双向BFS

例四

1. A*(曼哈顿距离)

2. A*(欧氏距离)

3. A*(切比雪夫距离)

4. Dijkstra

5. 双向BFS

A*启发式搜索基础的更多相关文章

  1. new 经典基础模板总结

    NOIP-NOI-ZJOI基础模板总结 目录 C++语言和STL库操作 重载运算符操作 /* 重载运算符 格式 如重载小于号 这里是以x递减为第一关键字比较,y递减为第二关键字比较 */ bool o ...

  2. 启发式搜索A-Star算法 【寻找 最短路径 算法】【地理几何位置 可利用的情况】

    在处理最短路径问题时,有一种启发式算法是我们应该了解的,由于其有着优秀的探索效率在各自现实项目中多有应用,它就是 A-star 算法,或  A*  算法. 个人观点: A*  算法并不保证找到的路径一 ...

  3. 启发式搜索技术A*

    开篇 这篇文章介绍找最短路径的一种算法,它的字我比较喜欢:启发式搜索. 对于入门的好文章不多,而这篇文章就是为初学者而写的,很适合入门的一篇.文章定位:非专业性A*文章,很适合入门. 有图有真相,先给 ...

  4. 基础路径规划算法(Dijikstra、A*、D*)总结

    引言 在一张固定地图上选择一条路径,当存在多条可选的路径之时,需要选择代价最小的那条路径.我们称这类问题为最短路径的选择问题.解决这个问题最经典的算法为Dijikstra算法,其通过贪心选择的步骤从源 ...

  5. java基础集合经典训练题

    第一题:要求产生10个随机的字符串,每一个字符串互相不重复,每一个字符串中组成的字符(a-zA-Z0-9)也不相同,每个字符串长度为10; 分析:*1.看到这个题目,或许你脑海中会想到很多方法,比如判 ...

  6. node-webkit 环境搭建与基础demo

    首先去github上面下载(地址),具体更具自己的系统,我的是windows,这里只给出windows的做法 下载windows x64版本 下载之后解压,得到以下东西 为了方便,我们直接在这个目录中 ...

  7. js学习笔记:webpack基础入门(一)

    之前听说过webpack,今天想正式的接触一下,先跟着webpack的官方用户指南走: 在这里有: 如何安装webpack 如何使用webpack 如何使用loader 如何使用webpack的开发者 ...

  8. Golang, 以17个简短代码片段,切底弄懂 channel 基础

    (原创出处为本博客:http://www.cnblogs.com/linguanh/) 前序: 因为打算自己搞个基于Golang的IM服务器,所以复习了下之前一直没怎么使用的协程.管道等高并发编程知识 ...

  9. [C#] C# 基础回顾 - 匿名方法

    C# 基础回顾 - 匿名方法 目录 简介 匿名方法的参数使用范围 委托示例 简介 在 C# 2.0 之前的版本中,我们创建委托的唯一形式 -- 命名方法. 而 C# 2.0 -- 引进了匿名方法,在 ...

随机推荐

  1. 【leetcode】519. Random Flip Matrix

    题目如下: You are given the number of rows n_rows and number of columns n_cols of a 2D binary matrix whe ...

  2. 重磅 | 阿里云与MongoDB达成战略合作,成为全球唯一提供最新版MongoDB的云厂商

    MongoDB是业界最受欢迎的开源数据库之一,2019年一份面向开发者的数据库调查报告中,MongoDB以 24.6%的使用率占据次席. 阿里云是国内最早提供MongoDB服务的云厂商,提供完全兼容M ...

  3. 常见sql操作

    1. select '`'||b.mrchno 商户号, b.name 商户名称, b.contact3 注册地址联系人, '`'||b.telno1 邮寄地址联系电话, a.MRCHT_NAME X ...

  4. webpack 兼容低版本浏览器,转换ES6 ES7语法

    ES6,ES7真的太棒了,async +await+Promise,让我阅读代码的时候不用再从左拉到右了(异步太多,一层套一层真的太头痛) 但是有个问题,打包后低版本浏览器运行不了,还有我用了一些混淆 ...

  5. JDK各个版本比较

    JDK5 自动装箱与拆箱: 枚举 静态导入,如:import staticjava.lang.System.out 可变参数(Varargs) 内省(Introspector) 主要用于操作JavaB ...

  6. gradle打成jar包报错 "错误: 找不到或无法加载主类 App"(已经配置过主类)

    文章目录 将gradle打成jar包(包括依赖) 运行jar包 报错 原因(src自己手动创建的) 解决(添加src目录) 将gradle打成jar包(包括依赖) jar { manifest { a ...

  7. Struts1.3——Struts标签

    1.struts标签的介绍 Struts框架提供了一组非常丰富的框架组件,同时也提供了一组标签库用于和这些组件交互,主要介绍以下三类: html标签 bean标签 logic标签 2.Html标签库 ...

  8. cesium清除选定事件

    cesium清除选定事件 此处的案例不一定适合你的项目,但可以给你一个思路.清除选定,就是还原你选中之前的状态.比如你点击一个面高亮,面的颜色发生改变:并且会弹出一个divPoint框.此时的清除选定 ...

  9. Redis Sentinel用法

    1 Redis Sentinel 1.1 哨兵的作用 1. 监控:监控主从是否正常 2. 通知:出现问题时,可以通知相关人员 3. 故障迁移:自动主从切换 4. 统一的配置管理:连接者询问sentin ...

  10. 转 Jmeter业务请求比例

    [转载]Jmeter业务请求比例1   ps:文章转自订阅号“测试那点事儿”,链接:https://mp.weixin.qq.com/s/qVD4iNO0QqRIwAIq9_E_Kw   方法二: 可 ...