标签:代码 简易 alt 运算 节点 出现 padding 简单的 etl
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 | 把起始格添加到 "开启列表"do{          寻找开启列表中F值最低的格子, 我们称它为当前格.          把它切换到关闭列表.          对当前格相邻的8格中的每一个             if(它不可通过 || 已经在 "关闭列表"中)             {                   什么也不做.              }             if(它不在开启列表中)             {                   把它添加进 "开启列表", 把当前格作为这一格的父节点, 计算这一格的 FGH             if(它已经在开启列表中)             {                   if(用G值为参考检查新的路径是否更好, 更低的G值意味着更好的路径)                       {                               把这一格的父节点改成当前格, 并且重新计算这一格的 GF 值.                       }   } while( 目标格已经在 "开启列表", 这时候路径被找到)   如果开启列表已经空了, 说明路径不存在.    最后从目标格开始, 沿着每一格的父节点移动直到回到起始格, 这就是路径.  | 
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 | #pragma once  /* //A*算法对象类 */#include <vector>  #include <list>    constintkCost1=10; //直移一格消耗  constintkCost2=14; //斜移一格消耗    structPoint  {      intx,y; //点坐标,这里为了方便按照C++的数组来计算,x代表横排,y代表竖列      intF,G,H; //F=G+H      Point *parent; //parent的坐标,这里没有用指针,从而简化代码      Point(int_x,int_y):x(_x),y(_y),F(0),G(0),H(0),parent(NULL)  //变量初始化      {      }  };      classAstar  {  public:      voidInitAstar(std::vector<std::vector<int>> &_maze);      std::list<Point *> GetPath(Point &startPoint,Point &endPoint,boolisIgnoreCorner);    private:      Point *findPath(Point &startPoint,Point &endPoint,boolisIgnoreCorner);      std::vector<Point *> getSurroundPoints(constPoint *point,boolisIgnoreCorner) const;      boolisCanreach(constPoint *point,constPoint *target,boolisIgnoreCorner) const; //判断某点是否可以用于下一步判断      Point *isInList(conststd::list<Point *> &list,constPoint *point) const; //判断开启/关闭列表中是否包含某点      Point *getLeastFpoint(); //从开启列表中返回F值最小的节点      //计算FGH值      intcalcG(Point *temp_start,Point *point);      intcalcH(Point *point,Point *end);      intcalcF(Point *point);  private:      std::vector<std::vector<int>> maze;      std::list<Point *> openList;  //开启列表      std::list<Point *> closeList; //关闭列表  };  | 
Astar.cpp
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 | #include <math.h>  #include "Astar.h"    voidAstar::InitAstar(std::vector<std::vector<int>> &_maze)  {      maze=_maze;  }    intAstar::calcG(Point *temp_start,Point *point)  {      intextraG=(abs(point->x-temp_start->x)+abs(point->y-temp_start->y))==1?kCost1:kCost2;      intparentG=point->parent==NULL?0:point->parent->G; //如果是初始节点,则其父节点是空      returnparentG+extraG;  }    intAstar::calcH(Point *point,Point *end)  {      //用简单的欧几里得距离计算H,这个H的计算是关键,还有很多算法,没深入研究^_^      returnsqrt((double)(end->x-point->x)*(double)(end->x-point->x)+(double)(end->y-point->y)*(double)(end->y-point->y))*kCost1;  }    intAstar::calcF(Point *point)  {      returnpoint->G+point->H;  }    Point *Astar::getLeastFpoint()  {      if(!openList.empty())      {          autoresPoint=openList.front();          for(auto&point:openList)              if(point->F<resPoint->F)                  resPoint=point;          returnresPoint;      }      returnNULL;  }    Point *Astar::findPath(Point &startPoint,Point &endPoint,boolisIgnoreCorner)  {      openList.push_back(newPoint(startPoint.x,startPoint.y)); //置入起点,拷贝开辟一个节点,内外隔离      while(!openList.empty())      {          autocurPoint=getLeastFpoint(); //找到F值最小的点          openList.remove(curPoint); //从开启列表中删除          closeList.push_back(curPoint); //放到关闭列表          //1,找到当前周围八个格中可以通过的格子          autosurroundPoints=getSurroundPoints(curPoint,isIgnoreCorner);          for(auto&target:surroundPoints)          {              //2,对某一个格子,如果它不在开启列表中,加入到开启列表,设置当前格为其父节点,计算F G H              if(!isInList(openList,target))              {                  target->parent=curPoint;                    target->G=calcG(curPoint,target);                  target->H=calcH(target,&endPoint);                  target->F=calcF(target);                    openList.push_back(target);              }              //3,对某一个格子,它在开启列表中,计算G值, 如果比原来的大, 就什么都不做, 否则设置它的父节点为当前点,并更新G和F              else            {                  inttempG=calcG(curPoint,target);                  if(tempG<target->G)                  {                      target->parent=curPoint;                        target->G=tempG;                      target->F=calcF(target);                  }              }              Point *resPoint=isInList(openList,&endPoint);              if(resPoint)                  returnresPoint; //返回列表里的节点指针,不要用原来传入的endpoint指针,因为发生了深拷贝          }      }        returnNULL;  }    std::list<Point *> Astar::GetPath(Point &startPoint,Point &endPoint,boolisIgnoreCorner)  {      Point *result=findPath(startPoint,endPoint,isIgnoreCorner);      std::list<Point *> path;      //返回路径,如果没找到路径,返回空链表      while(result)      {          path.push_front(result);          result=result->parent;      }      returnpath;  }    Point *Astar::isInList(conststd::list<Point *> &list,constPoint *point) const{      //判断某个节点是否在列表中,这里不能比较指针,因为每次加入列表是新开辟的节点,只能比较坐标      for(autop:list)          if(p->x==point->x&&p->y==point->y)              returnp;      returnNULL;  }    boolAstar::isCanreach(constPoint *point,constPoint *target,boolisIgnoreCorner) const{      if(target->x<0||target->x>maze.size()-1          ||target->y<0&&target->y>maze[0].size()-1          ||maze[target->x][target->y]==1          ||target->x==point->x&&target->y==point->y          ||isInList(closeList,target)) //如果点与当前节点重合、超出地图、是障碍物、或者在关闭列表中,返回false          returnfalse;      else    {          if(abs(point->x-target->x)+abs(point->y-target->y)==1) //非斜角可以              returntrue;          else        {              //斜对角要判断是否绊住              if(maze[point->x][target->y]==0&&maze[target->x][point->y]==0)                  returntrue;              else                returnisIgnoreCorner;          }      }  }    std::vector<Point *> Astar::getSurroundPoints(constPoint *point,boolisIgnoreCorner) const{      std::vector<Point *> surroundPoints;        for(intx=point->x-1;x<=point->x+1;x++)          for(inty=point->y-1;y<=point->y+1;y++)              if(isCanreach(point,newPoint(x,y),isIgnoreCorner))                  surroundPoints.push_back(newPoint(x,y));            returnsurroundPoints;  }   | 
main.cpp
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 | #include <iostream>  #include "Astar.h"  usingnamespacestd;    intmain()  {      //初始化地图,用二维矩阵代表地图,1表示障碍物,0表示可通      vector<vector<int>> maze={          {1,1,1,1,1,1,1,1,1,1,1,1},          {1,0,0,1,1,0,1,0,0,0,0,1},          {1,0,0,1,1,0,0,0,0,0,0,1},          {1,0,0,0,0,0,1,0,0,1,1,1},          {1,1,1,0,0,0,0,0,1,1,0,1},          {1,1,0,1,0,0,0,0,0,0,0,1},          {1,0,1,0,0,0,0,1,0,0,0,1},          {1,1,1,1,1,1,1,1,1,1,1,1}      };      Astar astar;      astar.InitAstar(maze);        //设置起始和结束点      Point start(1,1);      Point end(6,10);      //A*算法找寻路径      list<Point *> path=astar.GetPath(start,end,false);      //打印      for(auto&p:path)          cout<<‘(‘<<p->x<<‘,‘<<p->y<<‘)‘<<endl;        system("pause");      return0;   | 
标签:代码 简易 alt 运算 节点 出现 padding 简单的 etl
原文地址:https://www.cnblogs.com/long5683/p/11782375.html