`
jiang5495
  • 浏览: 94819 次
  • 性别: Icon_minigender_1
  • 来自: 湖南
社区版块
存档分类
最新评论

A*算法入门(转载)

阅读更多
   深入A*算法 
                                 -浅析A*算法在搜索最短路径中的应用

                                                  Sunway
  目 录
  1 A*算法的程序编写原理
  2 用A*算法实现最短路径的搜索


--------------------------------------------------------------------------------

  在这里我将对A*算法的实际应用进行一定的探讨,并且举一个有关A*算法在最短路径搜索的例子。值得注意的是这里并不对A*的基本的概念作介绍,如果你还对A*算法不清楚的话,请看姊妹篇《初识A*算法》。
  这里所举的例子是参考AMIT主页中的一个源程序,你可以在AMIT的站点上下载也可以在我的站点上下载。你使用这个源程序时,应该遵守一定的公约。

1、A*算法的程序编写原理
  我在《初识A*算法》中说过,A*算法是最好优先算法的一种。只是有一些约束条件而已。我们先来看看最好优先算法是如何编写的吧。如图有如下的状态空间:(起始位置是A,目标位置是P,字母后的数字表示节点的估价值)。
  如图有如下的状态空间:(起始位置是A,目标位置是P,字母后的数字表示节点的估价值)


图1 状态空间图

  搜索过程中设置两个表:OPEN和CLOSED。OPEN表保存了所有已生成而未考察的节点,CLOSED表中记录已访问过的节点。算法中有一步是根据估价函数重排OPEN表。这样循环中的每一步只考虑OPEN表中状态最好的节点。具体搜索过程如下:

  1)初始状态:
      OPEN=[A5];            CLOSED=[];
  2)估算A5,取得搜有子节点,并放入OPEN表中;
      OPEN=[B4, C4, D6];        CLOSED=[A5]
  3)估算B4,取得搜有子节点,并放入OPEN表中;
      OPEN=[C4, E5, F5, D6];      CLOSED=[B4, A5]
  4)估算C4;取得搜有子节点,并放入OPEN表中;
      OPEN=[H3, G4, E5, F5, D6]     CLOSED=[C4, B4, A5]
  5)估算H3,取得搜有子节点,并放入OPEN表中;
      OPEN=[O2, P3, G4, E5, F5, D6];  CLOSED=H3C4, B4, A5]
  6)估算O2,取得搜有子节点,并放入OPEN表中;
      OPEN=[P3, G4, E5, F5, D6];    CLOSED=[O2, H3, C4, B4, A5]
  7)估算P3,已得到解;

  看了具体的过程,再看看伪程序吧。算法的伪程序如下:

  Best_First_Search()
  {
    Open = [起始节点];
    Closed = [];
    while ( Open表非空 )
    {
      从Open中取得一个节点X, 并从OPEN表中删除.
      if (X是目标节点)
      {
        求得路径PATH;
        返回路径PATH;
      }
      for (每一个X的子节点Y)
      {
        if( Y不在OPEN表和CLOSE表中 )
        {
          求Y的估价值;
          并将Y插入OPEN表中; //还没有排序
        }
        else if( Y在OPEN表中 )
        {
          if( Y的估价值小于OPEN表的估价值 )
            更新OPEN表中的估价值;
        }
        else //Y在CLOSE表中
        {
          if( Y的估价值小于CLOSE表的估价值 )
          {
            更新CLOSE表中的估价值;
            从CLOSE表中移出节点, 并放入OPEN表中;
          }
        }
        将X节点插入CLOSE表中;
        按照估价值将OPEN表中的节点排序;
      } //end for
    } //end while
  } //end func

  啊!伪程序出来了,写一个源程序应该不是问题了,依葫芦画瓢就可以。A*算法的程序与此是一样的,只要注意估价函数中的g(n)的h(n)约束条件就可以了。不清楚的可以看看《初识A*算法》。好了,我们可以进入另一个重要的话题,用A*算法实现最短路径的搜索。在此之前你最好认真的理解前面的算法。不清楚可以找我。

2、用A*算法实现最短路径的搜索
  在游戏设计中,经常要涉及到最短路径的搜索,现在一个比较好的方法就是用A*算法进行设计。他的好处我们就不用管了,反正就是好!
  注意下面所说的都是以ClassAstar这个程序为蓝本,你可以在这里下载这个程序。这个程序是一个完整的工程。里面带了一个EXE文件。可以先看看。
  先复习一下,A*算法的核心是估价函数f(n),它包括g(n)和h(n)两部分。g(n)是已经走过的代价,h(n)是n到目标的估计代价。在这个例子中g(n)表示在状态空间从起始节点到n节点的 深度,h(n)表示n节点所在地图的位置到目标位置的直线距离。啊!一个是状态空间,一个是实际的地图,不要搞错了。再详细点说,有一个物体A,在地图上的坐标是(xa,ya),A所要到达的目标b的坐标是(xb,yb)。则开始搜索时,设置一个起始节点1,生成八个子节点2 - 9 因为有八个方向。如图:



图2 节点图


  先看搜索主函数:

  void AstarPathfinder::FindPath(int sx, int sy, int dx, int dy)
  {
    NODE *Node, *BestNode;
    int TileNumDest;

    //得到目标位置,作判断用
    TileNumDest = TileNum(sx, sy);

    //生成Open和Closed表
    OPEN=( NODE* )calloc(1,sizeof( NODE ));
    CLOSED=( NODE* )calloc(1,sizeof( NODE ));

    //生成起始节点,并放入Open表中
    Node=( NODE* )calloc(1,sizeof( NODE ));
    Node->g = 0;

    //这是计算h值
    Node->h = (dx-sx)*(dx-sx) + (dy-sy)*(dy-sy); // should really use sqrt().

    //这是计算f值,即估价值
    Node->f = Node->g+Node->h;
    Node->NodeNum = TileNum(dx, dy);
    Node->x = dx;
    Node->y = dy;

    OPEN->NextNode=Node; // make Open List point to first node
    for (;;)
    {
      //从Open表中取得一个估价值最好的节点
      BestNode=ReturnBestNode();

      //如果该节点是目标节点就退出
      if (BestNode->NodeNum == TileNumDest) // if we've found the end, break and finish
        break;
      //否则生成子节点
      GenerateSuccessors(BestNode,sx,sy);
    }
    PATH = BestNode;
  }

  再看看生成子节点函数 GenerateSuccessors:

  void AstarPathfinder::GenerateSuccessors(NODE *BestNode, int dx, int dy)
  {
    int x, y;

    //依次生成八个方向的子节点,简单!
    // Upper-Left
    if ( FreeTile(x=BestNode->x-TILESIZE, y=BestNode->y-TILESIZE) )
      GenerateSucc(BestNode,x,y,dx,dy);
    // Upper
    if ( FreeTile(x=BestNode->x, y=BestNode->y-TILESIZE) )
      GenerateSucc(BestNode,x,y,dx,dy);
    // Upper-Right
    if ( FreeTile(x=BestNode->x+TILESIZE, y=BestNode->y-TILESIZE) )
      GenerateSucc(BestNode,x,y,dx,dy);
    // Right
    if ( FreeTile(x=BestNode->x+TILESIZE, y=BestNode->y) )
      GenerateSucc(BestNode,x,y,dx,dy);
    // Lower-Right
    if ( FreeTile(x=BestNode->x+TILESIZE, y=BestNode->y+TILESIZE) )
      GenerateSucc(BestNode,x,y,dx,dy);
    // Lower
    if ( FreeTile(x=BestNode->x, y=BestNode->y+TILESIZE) )
      GenerateSucc(BestNode,x,y,dx,dy);
    // Lower-Left
    if ( FreeTile(x=BestNode->x-TILESIZE, y=BestNode->y+TILESIZE) )
      GenerateSucc(BestNode,x,y,dx,dy);
    // Left
    if ( FreeTile(x=BestNode->x-TILESIZE, y=BestNode->y) )
      GenerateSucc(BestNode,x,y,dx,dy);
  }

  看看最重要的函数GenerateSucc:

  void AstarPathfinder::GenerateSucc(NODE *BestNode,int x, int y, int dx, int dy)
  {
    int g, TileNumS, c = 0;
    NODE *Old, *Successor;

    //计算子节点的 g 值
    g = BestNode->g+1; // g(Successor)=g(BestNode)+cost of getting from BestNode to Successor
    TileNumS = TileNum(x,y); // identification purposes

    //子节点再Open表中吗?
    if ( (Old=CheckOPEN(TileNumS)) != NULL ) // if equal to NULL then not in OPEN list,
                         // else it returns the Node in Old
    {
      //若在
      for( c = 0; c <8; c++)
        if( BestNode->Child[c] == NULL ) // Add Old to the list of BestNode's Children
                         // (or Successors).
         break;
        BestNode->Child[c] = Old;
        //比较Open表中的估价值和当前的估价值(只要比较g值就可以了)
        if ( g g ) // if our new g value is Parent = BestNode;
          Old->g = g;
          Old->f = g + Old->h;
        }
      }
      else //在Closed表中吗?
        if ( (Old=CheckCLOSED(TileNumS)) != NULL ) // if equal to NULL then not in OPEN list
                              // else it returns the Node in Old
        {
          //若在
          for( c = 0; c<8; c++)
            if ( BestNode->Child[c] == NULL ) // Add Old to the list of BestNode's
                             // Children (or Successors). break;
            BestNode->Child[c] = Old;
            //比较Closed表中的估价值和当前的估价值(只要比较g值就可以了)
            if ( g g ) // if our new g value is Parent = BestNode;
              Old->g = g;
              Old->f = g + Old->h; //再依次更新Old的所有子节点的估价值
              PropagateDown(Old); // Since we changed the g value of Old, we need
                         // to propagate this new value downwards, i.e.
                         // do a Depth-First traversal of the tree!
             }
        }
        else //不在Open表中也不在Close表中
        {
          //生成新的节点
          Successor = ( NODE* )calloc(1,sizeof( NODE ));
          Successor->Parent = BestNode;
          Successor->g = g;
          Successor->h = (x-dx)*(x-dx) + (y-dy)*(y-dy); // should do sqrt(), but since we
                                   don't really
          Successor->f = g+Successor->h; // care about the distance but just which branch
          looks Successor->x = x; // better this should suffice. Anyayz it's faster.
          Successor->y = y;
          Successor->NodeNum = TileNumS;
          //再插入Open表中,同时排序。
          Insert(Successor); // Insert Successor on OPEN list wrt f
          for( c =0; c <8; c++)
            if ( BestNode->Child[c] == NULL ) // Add Old to the list of BestNode's
                              Children (or Successors).
            break;
          BestNode->Child[c] = Successor;
        }
  }

  哈哈。A*算法我懂了。当然,我希望你有这样的感觉。不过我还要再说几句。仔细看看这个程序,你会发现,这个程序和我前面说的伪程序有一些不同,在GenerateSucc函数中,当子节点在Closed表中时,没有将子节点从Closed表中删除并放入Open表中。而是直接的重新的计算该节点的所有子节点的估价值(用PropagateDown函数)。这样可以快一些。另当子节点在Open表和Closed表中时,重新的计算估价值后,没有重新的对Open表中的节点排序,我有些想不通,为什么不排呢?会不会是一个小小的BUG。你知道告诉我好吗?
  好了。主要的内容都讲完了,还是完整仔细的看看源程序吧。希望我所的对你有一点帮助,一点点也可以。如果你对文章中的观点有异议或有更好的解释都告诉我。
分享到:
评论

相关推荐

    A*算法的讲解PPT(A算法)

    A*算法是一种在图形搜索问题中寻找最优路径的启发式搜索算法,由Peter Hart、Nils Nilsson和Bertram Raphael在1968年提出。它的主要目标是在有限的计算时间内找到从起点到终点的最短路径,特别适用于静态路网中的...

    A*算法Matlab 代码

    A* 算法是一种在图形搜索中广泛使用的启发式搜索策略,它的主要目标是找到从起点到终点的最短路径。在这个特定的上下文中,我们讨论的是在Matlab环境中实现A*算法来解决路径规划问题。A*算法结合了Dijkstra算法的最...

    A星算法 c语言实现 a*算法

    A星(A*)算法是一种在图形搜索中广泛使用的路径规划算法,它的主要目标是找到从起点到终点的最短路径。A*算法结合了Dijkstra算法的全局最优性和最佳优先搜索的效率,通过引入启发式函数来指导搜索,使得算法能够更...

    基于matlab的双向A*算法

    A*算法(A-Star)是一种广泛应用的启发式搜索算法,它结合了Dijkstra算法的最优化路径寻找与启发式信息的高效性。而基于MATLAB实现的双向A*算法,则是A*算法的一种优化形式,旨在进一步提升路径规划的效率。 首先,...

    A*算法 matlab版

    A*算法,动态路径规划算法的一种,程序直接放到matlab即可运行。

    迷宫问题的A*算法(python实现)

    A*算法是一种高效的启发式搜索算法,广泛用于解决此类问题。本篇将详细介绍A*算法的原理以及其在Python中的实现,并结合附件中的代码和测试样例进行解析。 A*算法的核心思想是通过评估节点的启发式分数(f(n))来...

    A*算法解决十五数码问题(Python程序、报告)

    **A*算法** A*算法是一种启发式搜索算法,结合了Dijkstra算法的最优化路径搜索和贪婪最佳优先搜索的优点。它利用了一个评估函数f(n)来指导搜索,该函数由两部分组成:g(n)是从初始状态到当前节点的实际代价,h(n)是...

    a*算法流程图(只是流程图)

    a*算法流程图(只是流程图)A*算法是一种在静态路网中求解最短路径最有效的直接搜索方法,也是解决许多其他搜索问题的有效算法。算法中的距离估算值与实际值越接近,扩展的节点数越少, 搜索速度越快。

    A*走路 自动寻路A*算法 易语言源码优化版

    A*(A-star)算法是一种广泛应用的路径搜索算法,它在自动寻路系统中起着核心作用。易语言是一种中国本土开发的、面向对象的编程语言,它以其简单易学的特点受到很多程序员的喜爱。这个“A*走路 自动寻路A*算法 ...

    A*算法学习(python代码实现)

    A*算法是一种在图形搜索中用于寻找从起点到终点最短路径的有效算法,它结合了Dijkstra算法的全局最优性和Greedy最佳优先搜索算法的效率。A*算法的核心在于引入了启发式函数,使得搜索更加高效,能够在大量可能的路径...

    A*算法原版论文

    A* (A-Star)算法是一种静态路网中求解最短路径最有效的直接搜索方法,也是许多其他问题的常用启发式算法。注意——是最有效的直接搜索算法,之后涌现了很多预处理算法(如ALT,CH,HL等等),在线查询效率是A*算法...

    A*算法解决八数码问题(C++)

    A*算法是路径搜索领域中一种非常高效的启发式搜索算法,它在解决八数码问题时表现出色。八数码问题,又称滑动拼图游戏,是一个经典的计算机科学问题,玩家需要通过移动空格来重新排列一组数字,使得它们最终形成一个...

    人工智能_滑动积木块—A*算法

    《人工智能:滑动积木块与A*算法详解》 在人工智能领域,寻路算法是解决智能体在复杂环境中寻找最优路径的关键技术。其中,A*(A-Star)算法因其高效性和准确性,被广泛应用于游戏开发、地图导航、机器人路径规划等...

    A*算法入门

    本文旨在为完全没有背景知识的新手提供一份易于理解的入门指南,帮助大家掌握A*算法的基本原理。 #### 二、A*算法应用场景 在许多实际应用中,比如电子游戏中角色的自动寻路、地图应用中的驾车路线规划以及机器人...

    八数码问题A*算法代码

    《八数码问题与A*算法实现详解》 八数码问题,又称滑动拼图或15拼图,是一个经典的计算机科学问题,属于图灵完全问题的范畴。它涉及到在一个3x3的网格上,通过空格与其他数字进行交换,使得初始布局最终转变为预设...

    A*算法解决传教士与野人过河问题(可运行代码)

    ### A*算法解决传教士与野人过河问题 #### 概述 在计算机科学领域,特别是人工智能中,A*算法是一种广泛使用的路径搜索算法,它结合了最佳优先搜索和启发式方法来找到从起始节点到目标节点的最优路径。本文将详细...

    C语言实现:使用A*算法来解决15数码问题

    A*算法是一种预测算法,主要用于寻路等,根据当前状态和目标状态之间的差异,预测到达目标需要多少开销,根据这个开销来判断下一次选择以那个状态开始。这个开销在八数码问题中可以以路程为标准。

    8数码游戏 A*算法 C++实现

    在这个问题中,A*算法是一种广泛应用的搜索策略,它结合了最佳优先搜索和启发式搜索的特点,能有效地找到最短解路径。 A*算法的核心在于它使用一个评估函数来预测从当前状态到目标状态的预计成本。这个函数通常由两...

    算法分析A*算法完成拼图

    A*算法是一种在图形搜索问题中广泛使用的启发式搜索算法,它的主要目标是找到从起始节点到目标节点的最短路径。在这个场景中,"算法分析A*算法完成拼图"指的是使用A*算法来解决拼图游戏的问题,即如何高效地找到使...

    A*算法解决传教士—野人过河问题.zip

    A* 算法是一种广泛应用在路径搜索和图形遍历中的高效寻路算法,它结合了最佳优先搜索(如Dijkstra算法)和启发式搜索(如贪婪最佳优先搜索)的优点。在这个“A*算法解决传教士—野人过河问题”的实验中,我们将深入...

Global site tag (gtag.js) - Google Analytics