IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 人工智能 -> (十一)学习笔记autoware源码core_planning(astar_search) -> 正文阅读

[人工智能](十一)学习笔记autoware源码core_planning(astar_search)

#include "astar_search/astar_search.h"

AstarSearch::AstarSearch()
{
  ros::NodeHandle private_nh_("~");

  // base configs
  private_nh_.param<bool>("use_back", use_back_, true);
  private_nh_.param<bool>("use_potential_heuristic", use_potential_heuristic_, true);
  private_nh_.param<bool>("use_wavefront_heuristic", use_wavefront_heuristic_, false);
  private_nh_.param<double>("time_limit", time_limit_, 5000.0);

  // robot configs
  private_nh_.param<double>("robot_length", robot_length_, 4.5);
  private_nh_.param<double>("robot_width", robot_width_, 1.75);
  private_nh_.param<double>("robot_base2back", robot_base2back_, 1.0);
  private_nh_.param<double>("minimum_turning_radius", minimum_turning_radius_, 6.0);

  // search configs
  private_nh_.param<int>("theta_size", theta_size_, 48);
  private_nh_.param<double>("angle_goal_range", angle_goal_range_, 6.0);
  private_nh_.param<double>("curve_weight", curve_weight_, 1.2);
  private_nh_.param<double>("reverse_weight", reverse_weight_, 2.00);
  private_nh_.param<double>("lateral_goal_range", lateral_goal_range_, 0.5);
  private_nh_.param<double>("longitudinal_goal_range", longitudinal_goal_range_, 2.0);

  // costmap configs
  private_nh_.param<int>("obstacle_threshold", obstacle_threshold_, 100);
  private_nh_.param<double>("potential_weight", potential_weight_, 10.0);
  private_nh_.param<double>("distance_heuristic_weight", distance_heuristic_weight_, 1.0);

  createStateUpdateTable();
}

AstarSearch::~AstarSearch()
{
}

// state update table for hybrid astar
void AstarSearch::createStateUpdateTable()
{
  // Vehicle moving for each angle
  state_update_table_.resize(theta_size_);
  double dtheta = 2.0 * M_PI / theta_size_;

  // Minimum moving distance with one state update
  //     arc  = r                       * theta
  double step = minimum_turning_radius_ * dtheta;

  for (int i = 0; i < theta_size_; i++)
  {
    double theta = dtheta * i;

    // Calculate right and left circle
    // Robot moves along these circles
    double right_circle_center_x = minimum_turning_radius_ * std::sin(theta);
    double right_circle_center_y = minimum_turning_radius_ * -std::cos(theta);
    double left_circle_center_x = -right_circle_center_x;
    double left_circle_center_y = -right_circle_center_y;

    // Calculate x and y shift to next state
    NodeUpdate nu;

    // forward
    nu.shift_x = step * std::cos(theta);
    nu.shift_y = step * std::sin(theta);
    nu.rotation = 0.0;
    nu.index_theta = 0;
    nu.step = step;
    nu.curve = false;
    nu.back = false;
    state_update_table_[i].emplace_back(nu);

    // forward right
    nu.shift_x = right_circle_center_x + minimum_turning_radius_ * std::cos(M_PI_2 + theta - dtheta);
    nu.shift_y = right_circle_center_y + minimum_turning_radius_ * std::sin(M_PI_2 + theta - dtheta);
    nu.rotation = -dtheta;
    nu.index_theta = -1;
    nu.step = step;
    nu.curve = true;
    nu.back = false;
    state_update_table_[i].emplace_back(nu);

    // forward left
    nu.shift_x = left_circle_center_x + minimum_turning_radius_ * std::cos(-M_PI_2 + theta + dtheta);
    nu.shift_y = left_circle_center_y + minimum_turning_radius_ * std::sin(-M_PI_2 + theta + dtheta);
    nu.rotation = dtheta;
    nu.index_theta = 1;
    nu.step = step;
    nu.curve = true;
    nu.back = false;
    state_update_table_[i].emplace_back(nu);

    if (use_back_)
    {
      // backward
      nu.shift_x = step * std::cos(theta) * -1.0;
      nu.shift_y = step * std::sin(theta) * -1.0;
      nu.rotation = 0;
      nu.index_theta = 0;
      nu.step = step;
      nu.curve = false;
      nu.back = true;
      state_update_table_[i].emplace_back(nu);

      // backward right
      nu.shift_x = right_circle_center_x + minimum_turning_radius_ * std::cos(M_PI_2 + theta + dtheta);
      nu.shift_y = right_circle_center_y + minimum_turning_radius_ * std::sin(M_PI_2 + theta + dtheta);
      nu.rotation = dtheta;
      nu.index_theta = 1;
      nu.step = step;
      nu.curve = true;
      nu.back = true;
      state_update_table_[i].emplace_back(nu);

      // backward left
      nu.shift_x = left_circle_center_x + minimum_turning_radius_ * std::cos(-1.0 * M_PI_2 + theta - dtheta);
      nu.shift_y = left_circle_center_y + minimum_turning_radius_ * std::sin(-1.0 * M_PI_2 + theta - dtheta);
      nu.rotation = dtheta * -1.0;
      nu.index_theta = -1;
      nu.step = step;
      nu.curve = true;
      nu.back = true;
      state_update_table_[i].emplace_back(nu);
    }
  }
}

/*
initialize函数(源码略)用于初始化A*搜索的代价地图nodes_,nodes_的数据类型是std:vector<std::vector<std:vector>>,
可以形象化将nodes_类比为三维立方体,立方体内每个点都是AstarNode数据结构。
*/
void AstarSearch::initialize(const nav_msgs::OccupancyGrid& costmap)
{
  costmap_ = costmap;

  int height = costmap_.info.height;
  int width = costmap_.info.width;

  // size initialization
  nodes_.resize(height);
  for (int i = 0; i < height; i++)
  {
    nodes_[i].resize(width);
  }
  for (int i = 0; i < height; i++)
  {
    for (int j = 0; j < width; j++)
    {
      nodes_[i][j].resize(theta_size_);
    }
  }

  // cost initialization
  for (int i = 0; i < height; i++)
  {
    for (int j = 0; j < width; j++)
    {
      // Index of subscribing OccupancyGrid message
      int og_index = i * width + j;
      int cost = costmap_.data[og_index];

      // hc is set to be 0 when reset()
      if (cost == 0)
      {
        continue;
      }

      // obstacle or unknown area
      if (cost < 0 || obstacle_threshold_ <= cost)
      {
        nodes_[i][j][0].status = STATUS::OBS;
      }

      // the cost more than threshold is regarded almost same as an obstacle
      // because of its very high cost
      if (use_potential_heuristic_)
      {
        nodes_[i][j][0].hc = cost * potential_weight_;
      }
    }
  }
}

//makePlan函数设置搜索起点,基于wavefront算法更新nodes_中节点的启发式代价,最后调用search函数开启A*搜索。
bool AstarSearch::makePlan(const geometry_msgs::Pose& start_pose, const geometry_msgs::Pose& goal_pose)
{ 
  //设置起始节点
  if (!setStartNode(start_pose))
  {
    // ROS_WARN_STREAM("Invalid start pose");
    return false;
  }
  //基于wavefront算法更新nodes_中节点的启发式代价
  if (!setGoalNode(goal_pose))
  {
    // ROS_WARN_STREAM("Invalid goal pose");
    return false;
  }
  //开启A*搜索
  return search();
}

//setStartNode函数(源码略)设置起始地图节点,并将其加入 openList。
//poseToIndex函数根据传入函数的pose确定对应地图节点在三维栅格化代价地图nodes_中的坐标。
bool AstarSearch::setStartNode(const geometry_msgs::Pose& start_pose)
{
  // Get index of start pose
  int index_x, index_y, index_theta;
  start_pose_local_.pose = start_pose;
  poseToIndex(start_pose_local_.pose, &index_x, &index_y, &index_theta);
  SimpleNode start_sn(index_x, index_y, index_theta, 0, 0);

  // Check if start is valid
  if (isOutOfRange(index_x, index_y) || detectCollision(start_sn))
  {
    return false;
  }

  // Set start node
  AstarNode& start_node = nodes_[index_y][index_x][index_theta];
  start_node.x = start_pose_local_.pose.position.x;
  start_node.y = start_pose_local_.pose.position.y;
  start_node.theta = 2.0 * M_PI / theta_size_ * index_theta;
  start_node.gc = 0;
  start_node.move_distance = 0;
  start_node.back = false;
  start_node.status = STATUS::OPEN;
  start_node.parent = NULL;

  // set euclidean distance heuristic cost
  if (!use_wavefront_heuristic_ && !use_potential_heuristic_)
  {
    //calcDistance函数计算两点间平面距离。
    start_node.hc = calcDistance(start_pose_local_.pose.position.x, start_pose_local_.pose.position.y,
                                 goal_pose_local_.pose.position.x, goal_pose_local_.pose.position.y) *
                    distance_heuristic_weight_;
  }
  else if (use_potential_heuristic_)
  {
    start_node.gc += start_node.hc;
    start_node.hc += calcDistance(start_pose_local_.pose.position.x, start_pose_local_.pose.position.y,
                                  goal_pose_local_.pose.position.x, goal_pose_local_.pose.position.y) +
                     distance_heuristic_weight_;
  }

  // Push start node to openlist
  start_sn.cost = start_node.gc + start_node.hc;
  openlist_.push(start_sn);

  return true;
}

//setGoalNode函数的主要作用是从目标点开始基于wavefront算法更新nodes_中节点的启发式代价。
bool AstarSearch::setGoalNode(const geometry_msgs::Pose& goal_pose)
{
  goal_pose_local_.pose = goal_pose;

  //modifyTheta函数修正传入函数的角度theta,令其值处于[0,2pi)区间内。
  goal_yaw_ = modifyTheta(tf::getYaw(goal_pose_local_.pose.orientation));

  // Get index of goal pose
  //与setstartNode函数内流程基本一致
  int index_x, index_y, index_theta;
  poseToIndex(goal_pose_local_.pose, &index_x, &index_y, &index_theta);
  SimpleNode goal_sn(index_x, index_y, index_theta, 0, 0);

  // Check if goal is valid
  if (isOutOfRange(index_x, index_y) || detectCollision(goal_sn))
  {
    return false;
  }

  // Calculate wavefront heuristic cost
  //计算wavefront启发式成本
  if (use_wavefront_heuristic_)
  {
    // auto start = std::chrono::system_clock::now();
    //基于wavefront算法更新nodes_中节点的启发式代价
    bool wavefront_result = calcWaveFrontHeuristic(goal_sn);
    // auto end = std::chrono::system_clock::now();
    // auto usec = std::chrono::duration_cast<std::chrono::microseconds>(end - start).count();
    // std::cout << "wavefront : " << usec / 1000.0 << "[msec]" << std::endl;

    if (!wavefront_result)
    {
      // ROS_WARN("Reachable is false...");
      return false;
    }
  }

  return true;
}

//poseToIndex函数根据传入函数的pose确定对应地图节点在三维栅格化代价地图nodes_中的坐标。
void AstarSearch::poseToIndex(const geometry_msgs::Pose& pose, int* index_x, int* index_y, int* index_theta)
{
  tf::Transform orig_tf;
  tf::poseMsgToTF(costmap_.info.origin, orig_tf);
  geometry_msgs::Pose pose2d = transformPose(pose, orig_tf.inverse());

//根据横纵坐标的单位长度计算得到pose 对应的横纵坐标
  *index_x = pose2d.position.x / costmap_.info.resolution;
  *index_y = pose2d.position.y / costmap_.info.resolution;

  tf::Quaternion quat;
  tf::quaternionMsgToTF(pose2d.orientation, quat);
  double yaw = tf::getYaw(quat);
  if (yaw < 0)
    yaw += 2.0 * M_PI;

  // Descretize angle
  //获得角度这一维度坐标
  static double one_angle_range = 2.0 * M_PI / theta_size_;
  *index_theta = yaw / one_angle_range;
  *index_theta %= theta_size_;
}

void AstarSearch::pointToIndex(const geometry_msgs::Point& point, int* index_x, int* index_y)
{
  geometry_msgs::Pose pose;
  pose.position = point;
  int index_theta;
  poseToIndex(pose, index_x, index_y, &index_theta);
}

//isOutOfRange函数检查下标是否溢出。
bool AstarSearch::isOutOfRange(int index_x, int index_y)
{
  if (index_x < 0 || index_x >= static_cast<int>(costmap_.info.width) || index_y < 0 ||
      index_y >= static_cast<int>(costmap_.info.height))
  {
    return true;
  }
  return false;
}

//search函数启动基于A*算法的搜索过程。
bool AstarSearch::search()
{
  ros::WallTime begin = ros::WallTime::now();

  // Start A* search   开始A*搜索
  // If the openlist is empty, search failed  如果openList为空,搜索失败
  while (!openlist_.empty())
  {
    // Check time and terminate if the search reaches the time limit
    ros::WallTime now = ros::WallTime::now();
    double msec = (now - begin).toSec() * 1000.0;
    if (msec > time_limit_)
    {
      // ROS_WARN("Exceed time limit of %lf [ms]", time_limit_);
      return false;
    }

    // Pop minimum cost node from openlist
    //从openList弹出最低代价节点
    SimpleNode top_sn = openlist_.top();
    openlist_.pop();

    // Expand nodes from this node
    AstarNode* current_an = &nodes_[top_sn.index_y][top_sn.index_x][top_sn.index_theta];

    //将该节点状态设为STATUS ::CLOSED,即将该节点移入closeList
    current_an->status = STATUS::CLOSED;

    // Goal check  目标检查
    if (isGoal(current_an->x, current_an->y, current_an->theta))
    {
      // ROS_INFO("Search time: %lf [msec]", (now - begin).toSec() * 1000.0);
      setPath(top_sn);
      return true;
    }

    // Expand nodes  拓展节点
    for (const auto& state : state_update_table_[top_sn.index_theta])
    {
      // Next state 下一节点
      double next_x = current_an->x + state.shift_x;
      double next_y = current_an->y + state.shift_y;

      //modifyTheta函数修正传入函数的角度theta,令其值处于[0,2pi)区间内。
      double next_theta = modifyTheta(current_an->theta + state.rotation);
      double move_cost = state.step;
      double move_distance = current_an->move_distance + state.step;

      // Increase reverse cost
      //增加反向代价
      if (state.back != current_an->back)
        move_cost *= reverse_weight_;

      // Calculate index of the next state  计算下一个状态的下标
      SimpleNode next_sn;
      geometry_msgs::Point next_pos;
      next_pos.x = next_x;
      next_pos.y = next_y;
      pointToIndex(next_pos, &next_sn.index_x, &next_sn.index_y);
      next_sn.index_theta = top_sn.index_theta + state.index_theta;

      // Avoid invalid index  避免下标无效
      next_sn.index_theta = (next_sn.index_theta + theta_size_) % theta_size_;

      // Check if the index is valid  检查下标是否有效
      if (isOutOfRange(next_sn.index_x, next_sn.index_y) || detectCollision(next_sn))
      {
        continue;
      }

      AstarNode* next_an = &nodes_[next_sn.index_y][next_sn.index_x][next_sn.index_theta];
      double next_gc = current_an->gc + move_cost;

      //wavefront或距离变换启发式
      double next_hc = nodes_[next_sn.index_y][next_sn.index_x][0].hc;  // wavefront or distance transform heuristic

      // increase the cost with euclidean distance
      //增加欧氏距离成本
      if (use_potential_heuristic_)
      {
        next_gc += nodes_[next_sn.index_y][next_sn.index_x][0].hc;
        next_hc += calcDistance(next_x, next_y, goal_pose_local_.pose.position.x, goal_pose_local_.pose.position.y) *
                   distance_heuristic_weight_;
      }
      //calcDistance函数计算两点间平面距离。
      // increase the cost with euclidean distance
      if (!use_wavefront_heuristic_ && !use_potential_heuristic_)
      {
        next_hc = calcDistance(next_x, next_y, goal_pose_local_.pose.position.x, goal_pose_local_.pose.position.y) *
                  distance_heuristic_weight_;
      }

      // NONE
      if (next_an->status == STATUS::NONE)
      {
        next_an->status = STATUS::OPEN;
        next_an->x = next_x;
        next_an->y = next_y;
        next_an->theta = next_theta;
        next_an->gc = next_gc;
        next_an->hc = next_hc;
        next_an->move_distance = move_distance;
        next_an->back = state.back;
        next_an->parent = current_an;
        next_sn.cost = next_an->gc + next_an->hc;
        openlist_.push(next_sn);
        continue;
      }

      // OPEN or CLOSED
      if (next_an->status == STATUS::OPEN || next_an->status == STATUS::CLOSED)
      {
        if (next_gc < next_an->gc)
        {
          next_an->status = STATUS::OPEN;
          next_an->x = next_x;
          next_an->y = next_y;
          next_an->theta = next_theta;
          next_an->gc = next_gc;
          next_an->hc = next_hc;  // already calculated ?
          next_an->move_distance = move_distance;
          next_an->back = state.back;
          next_an->parent = current_an;
          next_sn.cost = next_an->gc + next_an->hc;
          openlist_.push(next_sn);
          continue;
        }
      }
    }  // state update   状态更新
  }

  // Failed to find path
  // ROS_INFO("Open list is empty...");
  return false;
}

//setPath函数设置从起始点到目标点的路径。
void AstarSearch::setPath(const SimpleNode& goal)
{
  std_msgs::Header header;
  header.stamp = ros::Time::now();
  header.frame_id = costmap_.header.frame_id;
  path_.header = header;

  // From the goal node to the start node
  AstarNode* node = &nodes_[goal.index_y][goal.index_x][goal.index_theta];

  while (node != NULL)
  {
    // Set tf pose
    tf::Vector3 origin(node->x, node->y, 0);
    tf::Pose tf_pose;
    tf_pose.setOrigin(origin);
    tf_pose.setRotation(tf::createQuaternionFromYaw(node->theta));

    // Set path as ros message
    geometry_msgs::PoseStamped ros_pose;
    tf::poseTFToMsg(tf_pose, ros_pose.pose);
    ros_pose.header = header;
    path_.poses.push_back(ros_pose);

    // To the next node
    node = node->parent;
  }

  // Reverse the vector to be start to goal order
  std::reverse(path_.poses.begin(), path_.poses.end());
}

// Check if the next state is the goal
// Check lateral offset, longitudinal offset and angle
// isGoal函数检查下一个状态是否是目标点。
bool AstarSearch::isGoal(double x, double y, double theta)
{
  // To reduce computation time, we use square value for distance
  static const double lateral_goal_range =
      lateral_goal_range_ / 2.0;  // [meter], divide by 2 means we check left and right
  static const double longitudinal_goal_range =
      longitudinal_goal_range_ / 2.0;                                         // [meter], check only behind of the goal
  static const double goal_angle = M_PI * (angle_goal_range_ / 2.0) / 180.0;  // degrees -> radian

  // Calculate the node coordinate seen from the goal point
  tf::Point p(x, y, 0);
  geometry_msgs::Point relative_node_point = calcRelativeCoordinate(goal_pose_local_.pose, p);

  // Check Pose of goal
  if (relative_node_point.x < 0 &&  // shoud be behind of goal
      std::fabs(relative_node_point.x) < longitudinal_goal_range &&
      std::fabs(relative_node_point.y) < lateral_goal_range)
  {
    // Check the orientation of goal
    if (calcDiffOfRadian(goal_yaw_, theta) < goal_angle)
    {
      return true;
    }
  }

  return false;
}

bool AstarSearch::isObs(int index_x, int index_y)
{
  if (nodes_[index_y][index_x][0].status == STATUS::OBS)
  {
    return true;
  }

  return false;
}

bool AstarSearch::detectCollision(const SimpleNode& sn)
{
  // Define the robot as rectangle
  static double left = -1.0 * robot_base2back_;
  static double right = robot_length_ - robot_base2back_;
  static double top = robot_width_ / 2.0;
  static double bottom = -1.0 * robot_width_ / 2.0;
  static double resolution = costmap_.info.resolution;

  // Coordinate of base_link in OccupancyGrid frame
  static double one_angle_range = 2.0 * M_PI / theta_size_;
  double base_x = sn.index_x * resolution;
  double base_y = sn.index_y * resolution;
  double base_theta = sn.index_theta * one_angle_range;

  // Calculate cos and sin in advance
  double cos_theta = std::cos(base_theta);
  double sin_theta = std::sin(base_theta);

  // Convert each point to index and check if the node is Obstacle
  for (double x = left; x < right; x += resolution)
  {
    for (double y = top; y > bottom; y -= resolution)
    {
      // 2D point rotation
      int index_x = (x * cos_theta - y * sin_theta + base_x) / resolution;
      int index_y = (x * sin_theta + y * cos_theta + base_y) / resolution;

      if (isOutOfRange(index_x, index_y))
      {
        return true;
      }
      else if (nodes_[index_y][index_x][0].status == STATUS::OBS)
      {
        return true;
      }
    }
  }

  return false;
}

bool AstarSearch::calcWaveFrontHeuristic(const SimpleNode& sn)
{
  // Set start point for wavefront search
  // This is goal for Astar search
  nodes_[sn.index_y][sn.index_x][0].hc = 0;
  WaveFrontNode wf_node(sn.index_x, sn.index_y, 1e-10);
  std::queue<WaveFrontNode> qu;
  qu.push(wf_node);

  // State update table for wavefront search
  // Nodes are expanded for each neighborhood cells (moore neighborhood)
  double resolution = costmap_.info.resolution;
  static std::vector<WaveFrontNode> updates = {
    getWaveFrontNode(0, 1, resolution),
    getWaveFrontNode(-1, 0, resolution),
    getWaveFrontNode(1, 0, resolution),
    getWaveFrontNode(0, -1, resolution),
    getWaveFrontNode(-1, 1, std::hypot(resolution, resolution)),
    getWaveFrontNode(1, 1, std::hypot(resolution, resolution)),
    getWaveFrontNode(-1, -1, std::hypot(resolution, resolution)),
    getWaveFrontNode(1, -1, std::hypot(resolution, resolution)),
  };

  // Get start index
  int start_index_x;
  int start_index_y;
  int start_index_theta;
  poseToIndex(start_pose_local_.pose, &start_index_x, &start_index_y, &start_index_theta);

  // Whether the robot can reach goal
  bool reachable = false;

  // Start wavefront search
  while (!qu.empty())
  {
    WaveFrontNode ref = qu.front();
    qu.pop();

    WaveFrontNode next;
    for (const auto& u : updates)
    {
      next.index_x = ref.index_x + u.index_x;
      next.index_y = ref.index_y + u.index_y;

      // out of range OR already visited OR obstacle node
      if (isOutOfRange(next.index_x, next.index_y) || nodes_[next.index_y][next.index_x][0].hc > 0 ||
          nodes_[next.index_y][next.index_x][0].status == STATUS::OBS)
      {
        continue;
      }

      // Take the size of robot into account
      if (detectCollisionWaveFront(next))
      {
        continue;
      }

      // Check if we can reach from start to goal
      if (next.index_x == start_index_x && next.index_y == start_index_y)
      {
        reachable = true;
      }

      // Set wavefront heuristic cost
      next.hc = ref.hc + u.hc;
      nodes_[next.index_y][next.index_x][0].hc = next.hc;

      qu.push(next);
    }
  }

  // End of search
  return reachable;
}

// Simple collidion detection for wavefront search
//detectCollision函数检查无人车占据的地图节点是否与障碍物所处的节点重合,以此判断车辆是否会碰撞。
bool AstarSearch::detectCollisionWaveFront(const WaveFrontNode& ref)
{
  // Define the robot as square
  static double half = robot_width_ / 2;
  double robot_x = ref.index_x * costmap_.info.resolution;
  double robot_y = ref.index_y * costmap_.info.resolution;

  for (double y = half; y > -1.0 * half; y -= costmap_.info.resolution)
  {
    for (double x = -1.0 * half; x < half; x += costmap_.info.resolution)
    {
      int index_x = (robot_x + x) / costmap_.info.resolution;
      int index_y = (robot_y + y) / costmap_.info.resolution;

      if (isOutOfRange(index_x, index_y))
      {
        return true;
      }

      if (nodes_[index_y][index_x][0].status == STATUS::OBS)
      {
        return true;
      }
    }
  }

  return false;
}

//reset函数用于重置astar_中用于A*搜索的信息。
void AstarSearch::reset()
{
  path_.poses.clear();

  // Clear queue
  std::priority_queue<SimpleNode, std::vector<SimpleNode>, std::greater<SimpleNode>> empty;
  std::swap(openlist_, empty);

  // ros::WallTime begin = ros::WallTime::now();

  // Reset node info here ...?
  for (size_t i = 0; i < costmap_.info.height; i++)
  {
    for (size_t j = 0; j < costmap_.info.width; j++)
    {
      for (int k = 0; k < theta_size_; k++)
      {
        // other values will be updated during the search
        nodes_[i][j][k].status = STATUS::NONE;
        nodes_[i][j][k].hc = 0;
      }
    }
  }

  // ros::WallTime end = ros::WallTime::now();

  // ROS_INFO("Reset time: %lf [ms]", (end - begin).toSec() * 1000);
}

参考书目《Autoware与自动驾驶技术》

注:可以下载Visual Studio Coded代码编辑软件,展示性好点。

  人工智能 最新文章
2022吴恩达机器学习课程——第二课(神经网
第十五章 规则学习
FixMatch: Simplifying Semi-Supervised Le
数据挖掘Java——Kmeans算法的实现
大脑皮层的分割方法
【翻译】GPT-3是如何工作的
论文笔记:TEACHTEXT: CrossModal Generaliz
python从零学(六)
详解Python 3.x 导入(import)
【答读者问27】backtrader不支持最新版本的
上一篇文章      下一篇文章      查看所有文章
加:2022-01-29 23:05:20  更:2022-01-29 23:07:13 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/10 15:57:30-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码