Apollo星火计划学习笔记——Control 专项讲解(LQR)

news2025/1/10 14:18:16

文章目录

  • 1. 算法相关基础
    • 1.1 一阶倒立摆
    • 1.2 二自由度车辆横向跟踪偏差模型
    • 1.3 车辆横向跟踪偏差模型
    • 1.4 车辆横向跟踪偏差倒车模型
    • 1.5 轮胎侧偏角与侧偏刚度
    • 1.6 LQR 线性二次型问题:
  • 2. LQR代码解析
    • 2.1 WriteHeaders(调试过程中的状态量)
    • 2.2 LatController::LoadControlConf(读取横向配置中的参数)
    • 2.3 LatController::InitializeFilters(低通滤波器)
    • 2.4 LatController::Init(横向控制器的初始化)
    • 2.5 LatController::LoadLatGainScheduler(横向增益的加载)
    • 2.6 LatController::ComputeControlCommand(横向控制指令)
    • 2.7 LatController::UpdateDrivingOrientation()
    • 2.8 LatController::UpdateState
    • 2.9 LatController::ComputeLateralErrors
  • 3. 云平台实践教学
    • 3.1 前置步骤
    • 3.2 控制参数修改

1. 算法相关基础

1.1 一阶倒立摆

    一阶倒立摆类似于车辆横向控制。
在这里插入图片描述
    一阶倒立摆的相关内容可以参考以下两篇博客:
直线一级倒立摆数学建模与控制仿真
线性系统大作业——1.一阶倒立摆建模与控制系统设计
    一阶倒立摆是一个非线性的控制问题,对其线性化后,通过调整零极点可以增强其稳定性。在车辆的控制上,可以通过状态空间的形式来表示车辆变化的过程。
    对于一般的状态问题,我们给予以下表述: x ˙ = A x + B u \dot x=Ax+Bu x˙=Ax+Bu      x x x在作用 u u u下,会形成一个新的状态 x ˙ \dot x x˙.
    在Apollo代码中,将 x x x描述为偏差的形式,包括横向偏差、横向偏差的变化率以及轨迹朝向的偏差与变化率。 e ˙ r r = A e r r + B u \dot err=Aerr+Bu e˙rr=Aerr+Bu

1.2 二自由度车辆横向跟踪偏差模型

在这里插入图片描述在这里插入图片描述
在这里插入图片描述
    车辆动力学及控制的百度盘链接,想要进一步了解的读者可以下载。
链接:https://pan.baidu.com/s/14ScrNy6aho5vAIgRVEl9bQ?pwd=pk0x
提取码:pk0x
在这里插入图片描述

1.3 车辆横向跟踪偏差模型

在这里插入图片描述

1.4 车辆横向跟踪偏差倒车模型

在这里插入图片描述

1.5 轮胎侧偏角与侧偏刚度

在这里插入图片描述

在这里插入图片描述

    注意:此处需要将 N/°转化为N/rad
    图中可以看出7000N载荷时侧偏刚度大约为 -1000N/°,转化后为-57295.78N/rad,所以apollo中155494.663为2倍结果,也就是两轮之和。
    侧偏角与侧片偏刚度使用左手系,符号有出入。

1.6 LQR 线性二次型问题:

    对于给定的一个系统,已知代价函数函数cost function,能否求解出一个控制序列使得整个控制系统稳定且使代价函数值达到最小。
    对于一个线性离散系统 x k + 1 = A x k + B u k {x_{k + 1}} = A{x_k} + B{u_k} xk+1=Axk+Buk    其拥有以下代价函数: J = x N T Q x N + ∑ k = 0 N − 1 ( x k T Q x k + u k T R u k ) J = x_N^TQ{x_N} + \sum\limits_{k = 0}^{N - 1} {(x_k^TQ{x_k} + u_k^TR{u_k})} J=xNTQxN+k=0N1(xkTQxk+ukTRuk)    LQR 线性二次型问题就是求解最优控制序列 u u u,使得代价函数 J J J达到最小.

L L L——线性系统,若非线性则需将其线性化
Q Q Q——代价函数,限定其为二次型
R R R——调节器
L Q R LQR LQR目的——既能满足控制目标的要求,又能使输入能量达到最小。 ∑ k = 0 N − 1 ( x k T Q x k + u k T R u k ) \sum\limits_{k = 0}^{N - 1} {(x_k^TQ{x_k} + u_k^TR{u_k})} k=0N1(xkTQxk+ukTRuk)    通过累计求和,使得代价函数 J J J达到一个最小值,当 N N N趋向于无穷时,状态量 x k x_k xk会趋向于0,即系统趋于稳定,同时由于系统趋于稳定, u k u_k uk也会趋于0.也就是说,随着时间推移,我们需要对系统施加的控制量就会越来越小。
    求解代价函数 J J J最小值的方法有哈密尔顿方程拉格朗日法等。
    无论是哪种求解方法,最终的目的是求解出最优的状态反馈控制器: u k = − K k x k {u_k} = - {K_k}{x_k} uk=Kkxk    其中反馈增益矩阵K为: K k = ( R + B T P k + 1 B ) − 1 ( B T P k + 1 A + N T ) {K_k} = {(R + {B^T}{P_{k + 1}}B)^{ - 1}}({B^T}{P_{k + 1}}A + {N^T}) Kk=(R+BTPk+1B)1(BTPk+1A+NT)    通过 K K K我们可以求解出在当前状态下,输出的控制量的大小。
    上式中, P P P为连续时间黎卡提方程的解,可以通过迭代求解得到: P k − 1 = A T P k A − ( A T P k B + N ) ( R + B T P k B ) − 1 ( B T P k A + N T ) + Q {P_{k - 1}} = {A^T}{P_k}A - ({A^T}{P_k}B + N){(R + {B^T}{P_k}B)^{ - 1}}({B^T}{P_k}A + {N^T}) + Q Pk1=ATPkA(ATPkB+N)(R+BTPkB)1(BTPkA+NT)+Q注意:从终端条件 P N = Q P_N=Q PN=Q 开始迭代


    基于偏差的形式构建状态转移方程为 e r r k + 1 = A e r r k + B u k + C θ ˙ r er{r_{k + 1}} = Aer{r_k} + B{u_k} + C{\dot \theta _r} errk+1=Aerrk+Buk+Cθ˙r    在代码中,黎卡提方程 P k − 1 = A T P k A − ( A T P k B + N ) ( R + B T P k B ) − 1 ( B T P k A + N T ) + Q {P_{k - 1}} = {A^T}{P_k}A - ({A^T}{P_k}B + N){(R + {B^T}{P_k}B)^{ - 1}}({B^T}{P_k}A + {N^T}) + Q Pk1=ATPkA(ATPkB+N)(R+BTPkB)1(BTPkA+NT)+Q如下图所示
在这里插入图片描述
    其中 K k = ( R + B T P k + 1 B ) − 1 ( B T P k + 1 A + N T ) {K_k} = {(R + {B^T}{P_{k + 1}}B)^{ - 1}}({B^T}{P_{k + 1}}A + {N^T}) Kk=(R+BTPk+1B)1(BTPk+1A+NT)表示如下:在这里插入图片描述


    我们使用LQR计算出的控制量 u k = − K k ∗ e r r k {u_k} = - {K_k}*{err_k} uk=Kkerrk,难以将车辆维持在车道中心线上,实际上,道路存在的曲率等特征会影响到系统,从而产生一定的稳态误差 e r r k + 1 = A e r r k + B u k + C θ ˙ r er{r_{k + 1}} = Aer{r_k} + B{u_k} + C{\dot \theta _r} errk+1=Aerrk+Buk+Cθ˙r
    此时我们引入前馈控制量 δ f \delta_f δf 作用到输出上,减少控制系统的稳态误差 u k = − K k ∗ e r r k + δ f {u_k} = - {K_k}*{err_k}+\delta_f uk=Kkerrk+δf    车辆横向跟踪偏差模型如下所示:在这里插入图片描述
    《车辆动力学及控制》第三章中提出稳态误差求解公式:在这里插入图片描述
    考虑侧偏刚度符号问题,适配到Apollo动力学模型得到如下公式及实现:在这里插入图片描述
在这里插入图片描述
    其中kv为不足转向斜率.

2. LQR代码解析

2.1 WriteHeaders(调试过程中的状态量)

//保存到日志中的状态量
void WriteHeaders(std::ofstream &file_stream) {
  file_stream << "current_lateral_error,"       			//当前横向偏差
              << "current_ref_heading,"						//当前参考航向
              << "current_heading,"							//当前航向
              << "current_heading_error,"					//当前航向偏差
              << "heading_error_rate,"						//航向误差变化率
              << "lateral_error_rate,"						//横向误差变化率
	          << "current_curvature,"						//当前曲率
              << "steer_angle,"								//方向盘转角
              << "steer_angle_feedforward,"					//前馈控制计算输出
              << "steer_angle_lateral_contribution,"		//k矩阵中横向偏差计算结果分量
              << "steer_angle_lateral_rate_contribution,"	//k矩阵中横向偏差率计算结果分量
              << "steer_angle_heading_contribution,"		//k矩阵中航向偏差计算结果分量
              << "steer_angle_heading_rate_contribution,"	//k矩阵中航向偏差率计算结果分量
              << "steer_angle_feedback,"					//LQR计算出的方向盘控制量
              << "steering_position,"						//当前底盘方向盘状态
              << "v" << std::endl;							//车速
}

    控制结果未到达期望,可以首先查看当前底盘方向盘状态steering_position与方向盘转角的输出steer_angle是否吻合。若吻合(两者相近),则说明控制的输出量偏小,可以通过适当调节横向偏差及其变化率与航向偏差及其变化率(调节的是权重)。再者,可以参考k矩阵相关的四个参数。

2.2 LatController::LoadControlConf(读取横向配置中的参数)

// 读取横向配置中的参数
bool LatController::LoadControlConf(const ControlConf *control_conf) {
  if (!control_conf) {
    AERROR << "[LatController] control_conf == nullptr";
    return false;
  }
  vehicle_param_ =
      common::VehicleConfigHelper::Instance()->GetConfig().vehicle_param(); // 加载车辆配置参数

  ts_ = control_conf->lat_controller_conf().ts();							// 异常检查,若ts小于0,则为无效的时间间隔
  if (ts_ <= 0.0) {
    AERROR << "[MPCController] Invalid control update interval.";
    return false;
  }
  cf_ = control_conf->lat_controller_conf().cf();							// 前两轮轮胎侧偏刚度
  cr_ = control_conf->lat_controller_conf().cr();							// 后两轮轮胎侧偏刚度
  preview_window_ = control_conf->lat_controller_conf().preview_window();
  lookahead_station_low_speed_ =											// 低速前行预瞄距离,修复不同速度下的参考点的对准问题											
      control_conf->lat_controller_conf().lookahead_station();
  lookback_station_low_speed_ =												// 低速倒车预瞄距离
      control_conf->lat_controller_conf().lookback_station();
  lookahead_station_high_speed_ =											// 高速倒车预瞄距离
      control_conf->lat_controller_conf().lookahead_station_high_speed();
  lookback_station_high_speed_ =											// 高速前行预瞄距离
      control_conf->lat_controller_conf().lookback_station_high_speed();
  wheelbase_ = vehicle_param_.wheel_base();									// 轴距
  steer_ratio_ = vehicle_param_.steer_ratio();								// 转向比例
  steer_single_direction_max_degree_ =										// 最大方向转角
      vehicle_param_.max_steer_angle() / M_PI * 180;
  max_lat_acc_ = control_conf->lat_controller_conf().max_lateral_acceleration();// 最大侧向加速度
  low_speed_bound_ = control_conf_->lon_controller_conf().switch_speed();	// 横向配置低速时的速度约束
  low_speed_window_ =
      control_conf_->lon_controller_conf().switch_speed_window();

  const double mass_fl = control_conf->lat_controller_conf().mass_fl();		// 车辆左前质量
  const double mass_fr = control_conf->lat_controller_conf().mass_fr();
  const double mass_rl = control_conf->lat_controller_conf().mass_rl();
  const double mass_rr = control_conf->lat_controller_conf().mass_rr();
  const double mass_front = mass_fl + mass_fr;								// 车辆前悬质量
  const double mass_rear = mass_rl + mass_rr;
  mass_ = mass_front + mass_rear;											// 车辆总质量

  lf_ = wheelbase_ * (1.0 - mass_front / mass_);							// 车辆前悬长度
  lr_ = wheelbase_ * (1.0 - mass_rear / mass_);								// 车辆后悬长度

  // moment of inertia
  iz_ = lf_ * lf_ * mass_front + lr_ * lr_ * mass_rear;						// 车辆转动惯量

  lqr_eps_ = control_conf->lat_controller_conf().eps();						// LQR优化的最小精度0.01,停止条件为1
  lqr_max_iteration_ = control_conf->lat_controller_conf().max_iteration(); // LQR最大迭代次数

  query_relative_time_ = control_conf->query_relative_time();				// 相对时间,机构延时(大约500ms的延时)

  minimum_speed_protection_ = control_conf->minimum_speed_protection();		// 最小速度

  return true;
}

2.3 LatController::InitializeFilters(低通滤波器)

// 低通滤波器 去除高频噪声 效果同积分器
void LatController::InitializeFilters(const ControlConf *control_conf) {
  // Low pass filter
  std::vector<double> den(3, 0.0);
  std::vector<double> num(3, 0.0);
  common::LpfCoefficients(
      ts_, control_conf->lat_controller_conf().cutoff_freq(), &den, &num);
  digital_filter_.set_coefficients(den, num);
  // 偏差均值滤波 防止突变
  lateral_error_filter_ = common::MeanFilter(static_cast<std::uint_fast8_t>(
      control_conf->lat_controller_conf().mean_filter_window_size()));
  heading_error_filter_ = common::MeanFilter(static_cast<std::uint_fast8_t>(
      control_conf->lat_controller_conf().mean_filter_window_size()));
}

2.4 LatController::Init(横向控制器的初始化)

// 控制器初始化,动力学模型初始化
Status LatController::Init(std::shared_ptr<DependencyInjector> injector,
                           const ControlConf *control_conf) {
  control_conf_ = control_conf;
  injector_ = injector;
  if (!LoadControlConf(control_conf_)) {
    AERROR << "failed to load control conf";
    return Status(ErrorCode::CONTROL_COMPUTE_ERROR,
                  "failed to load control_conf");
  }
  // Matrix init operations.
  const int matrix_size = basic_state_size_ + preview_window_;		// 矩阵大小默认为基本状态大小 4维
  matrix_a_ = Matrix::Zero(basic_state_size_, basic_state_size_);	// 矩阵a常量
  matrix_ad_ = Matrix::Zero(basic_state_size_, basic_state_size_);	// 离散化矩阵A
  matrix_adc_ = Matrix::Zero(matrix_size, matrix_size);
  /*
  A matrix (Gear Drive)
  [0.0, 1.0, 0.0, 0.0;
   0.0, (-(c_f + c_r) / m) / v, (c_f + c_r) / m,
   (l_r * c_r - l_f * c_f) / m / v;
   0.0, 0.0, 0.0, 1.0;
   0.0, ((lr * cr - lf * cf) / i_z) / v, (l_f * c_f - l_r * c_r) / i_z,
   (-1.0 * (l_f^2 * c_f + l_r^2 * c_r) / i_z) / v;]
  */
  // 矩阵a速度无关项
  matrix_a_(0, 1) = 1.0;
  matrix_a_(1, 2) = (cf_ + cr_) / mass_;
  matrix_a_(2, 3) = 1.0;
  matrix_a_(3, 2) = (lf_ * cf_ - lr_ * cr_) / iz_;
  // 矩阵a速度相关项
  matrix_a_coeff_ = Matrix::Zero(matrix_size, matrix_size);
  matrix_a_coeff_(1, 1) = -(cf_ + cr_) / mass_;
  matrix_a_coeff_(1, 3) = (lr_ * cr_ - lf_ * cf_) / mass_;
  matrix_a_coeff_(3, 1) = (lr_ * cr_ - lf_ * cf_) / iz_;
  matrix_a_coeff_(3, 3) = -1.0 * (lf_ * lf_ * cf_ + lr_ * lr_ * cr_) / iz_;

  /*
  b = [0.0, c_f / m, 0.0, l_f * c_f / i_z]^T
  */
  matrix_b_ = Matrix::Zero(basic_state_size_, 1);
  matrix_bd_ = Matrix::Zero(basic_state_size_, 1);
  matrix_bdc_ = Matrix::Zero(matrix_size, 1);
  // 矩阵b
  matrix_b_(1, 0) = cf_ / mass_;
  matrix_b_(3, 0) = lf_ * cf_ / iz_;
  matrix_bd_ = matrix_b_ * ts_;
 // LQR权重矩阵
  matrix_state_ = Matrix::Zero(matrix_size, 1);
  matrix_k_ = Matrix::Zero(1, matrix_size); //反馈矩阵
  matrix_r_ = Matrix::Identity(1, 1);
  matrix_q_ = Matrix::Zero(matrix_size, matrix_size);

  int q_param_size = control_conf_->lat_controller_conf().matrix_q_size();
  int reverse_q_param_size =
      control_conf_->lat_controller_conf().reverse_matrix_q_size();
  if (matrix_size != q_param_size || matrix_size != reverse_q_param_size) {
    const auto error_msg = absl::StrCat(
        "lateral controller error: matrix_q size: ", q_param_size,
        "lateral controller error: reverse_matrix_q size: ",
        reverse_q_param_size,
        " in parameter file not equal to matrix_size: ", matrix_size);
    AERROR << error_msg;
    return Status(ErrorCode::CONTROL_COMPUTE_ERROR, error_msg);
  }

  for (int i = 0; i < q_param_size; ++i) {
    matrix_q_(i, i) = control_conf_->lat_controller_conf().matrix_q(i);
  }

  matrix_q_updated_ = matrix_q_;
  InitializeFilters(control_conf_);									// 初始化滤波器 均值滤波 低通数字滤波器
  auto &lat_controller_conf = control_conf_->lat_controller_conf();
  LoadLatGainScheduler(lat_controller_conf);
  LogInitParameters();
  // 滞后超前控制器
  enable_leadlag_ = control_conf_->lat_controller_conf()
                        .enable_reverse_leadlag_compensation();
  if (enable_leadlag_) {
    leadlag_controller_.Init(lat_controller_conf.reverse_leadlag_conf(), ts_);
  }
  // 模型参考自适应控制
  enable_mrac_ =
      control_conf_->lat_controller_conf().enable_steer_mrac_control();
  if (enable_mrac_) {
    mrac_controller_.Init(lat_controller_conf.steer_mrac_conf(),
                          vehicle_param_.steering_latency_param(), ts_);
  }

  enable_look_ahead_back_control_ =
      control_conf_->lat_controller_conf().enable_look_ahead_back_control();

  return Status::OK();
}

2.5 LatController::LoadLatGainScheduler(横向增益的加载)

void LatController::LoadLatGainScheduler(
    const LatControllerConf &lat_controller_conf) {
  const auto &lat_err_gain_scheduler =										// 不同速度下的横向偏差增益
      lat_controller_conf.lat_err_gain_scheduler();
  const auto &heading_err_gain_scheduler =									// 不同速度下的航向偏差增益
      lat_controller_conf.heading_err_gain_scheduler();
  AINFO << "Lateral control gain scheduler loaded";
  Interpolation1D::DataType xy1, xy2;										// 加载标定表
  for (const auto &scheduler : lat_err_gain_scheduler.scheduler()) {
    xy1.push_back(std::make_pair(scheduler.speed(), scheduler.ratio()));	// 横向偏差标定表
  }
  for (const auto &scheduler : heading_err_gain_scheduler.scheduler()) {
    xy2.push_back(std::make_pair(scheduler.speed(), scheduler.ratio()));	// 航向偏差标定表
  }

  lat_err_interpolation_.reset(new Interpolation1D);						// 一维线性插值
  ACHECK(lat_err_interpolation_->Init(xy1))
      << "Fail to load lateral error gain scheduler";

  heading_err_interpolation_.reset(new Interpolation1D);
  ACHECK(heading_err_interpolation_->Init(xy2))
      << "Fail to load heading error gain scheduler";
}

2.6 LatController::ComputeControlCommand(横向控制指令)

Status LatController::ComputeControlCommand(
    const localization::LocalizationEstimate *localization,
    const canbus::Chassis *chassis,
    const planning::ADCTrajectory *planning_published_trajectory,
    ControlCommand *cmd) {
  auto vehicle_state = injector_->vehicle_state();

  auto target_tracking_trajectory = *planning_published_trajectory;
  // Transform the coordinate of the planning trajectory from the center of the
  // rear-axis to the center of mass, if conditions matched
  // 低速时使用几何中心
  if (((FLAGS_trajectory_transform_to_com_reverse &&
        vehicle_state->gear() == canbus::Chassis::GEAR_REVERSE) ||
       (FLAGS_trajectory_transform_to_com_drive &&
        vehicle_state->gear() == canbus::Chassis::GEAR_DRIVE)) &&
      enable_look_ahead_back_control_) {
    trajectory_analyzer_.TrajectoryTransformToCOM(lr_);
  }

    对前进或倒车时中点的选择,倒车时会使用车辆的后轴中心,前进时会使用车辆的几何中心。倒车时主要使用的是车辆的运动学模型。

  // Re-build the vehicle dynamic models at reverse driving (in particular,
  // replace the lateral translational motion dynamics with the corresponding
  // kinematic models)
  if (vehicle_state->gear() == canbus::Chassis::GEAR_REVERSE) {
    /*
    A matrix (Gear Reverse)
    [0.0, 0.0, 1.0 * v 0.0;
     0.0, (-(c_f + c_r) / m) / v, (c_f + c_r) / m,
     (l_r * c_r - l_f * c_f) / m / v;
     0.0, 0.0, 0.0, 1.0;
     0.0, ((lr * cr - lf * cf) / i_z) / v, (l_f * c_f - l_r * c_r) / i_z,
     (-1.0 * (l_f^2 * c_f + l_r^2 * c_r) / i_z) / v;]
    */
    cf_ = -control_conf_->lat_controller_conf().cf();
    cr_ = -control_conf_->lat_controller_conf().cr();
    matrix_a_(0, 1) = 0.0;
    matrix_a_coeff_(0, 2) = 1.0;
  } else {

    对矩阵a的个别项进行修改。

2.7 LatController::UpdateDrivingOrientation()

void LatController::UpdateDrivingOrientation() {
  auto vehicle_state = injector_->vehicle_state();
  driving_orientation_ = vehicle_state->heading();
  matrix_bd_ = matrix_b_ * ts_;
  // Reverse the driving direction if the vehicle is in reverse mode
  if (FLAGS_reverse_heading_control) {
    if (vehicle_state->gear() == canbus::Chassis::GEAR_REVERSE) {
      driving_orientation_ =
          common::math::NormalizeAngle(driving_orientation_ + M_PI);
      // Update Matrix_b for reverse mode
      matrix_bd_ = -matrix_b_ * ts_;
      ADEBUG << "Matrix_b changed due to gear direction";
    }
  }
}

    对驾驶车辆的运动方向做判断,同时更新控制结果的输出。

 // Update state = [Lateral Error, Lateral Error Rate, Heading Error, Heading
  // Error Rate, preview lateral error1 , preview lateral error2, ...]
  UpdateState(debug);

  UpdateMatrix(); //矩阵状态更新

  // Compound discrete matrix with road preview model
  UpdateMatrixCompound();

    对横向偏差、横向偏差变化率、航向偏差、航向偏差变化率进行计算。

2.8 LatController::UpdateState

void LatController::UpdateState(SimpleLateralDebug *debug) {
  auto vehicle_state = injector_->vehicle_state();
  if (FLAGS_use_navigation_mode) {
    ComputeLateralErrors(
        0.0, 0.0, driving_orientation_, vehicle_state->linear_velocity(),
        vehicle_state->angular_velocity(), vehicle_state->linear_acceleration(),
        trajectory_analyzer_, debug);
  } else {
    // Transform the coordinate of the vehicle states from the center of the
    // rear-axis to the center of mass, if conditions matched
    const auto &com = vehicle_state->ComputeCOMPosition(lr_);
    ComputeLateralErrors(
        com.x(), com.y(), driving_orientation_,
        vehicle_state->linear_velocity(), vehicle_state->angular_velocity(),
        vehicle_state->linear_acceleration(), trajectory_analyzer_, debug);
  }

    ComputeLateralErrors通过获取当前车辆的位置、车辆的行驶状态(前进还是后退)、车辆的线速度、角速度、线加速度、轨迹等信息来计算横向偏差

2.9 LatController::ComputeLateralErrors

void LatController::ComputeLateralErrors(
    const double x, const double y, const double theta, const double linear_v,
    const double angular_v, const double linear_a,
    const TrajectoryAnalyzer &trajectory_analyzer, SimpleLateralDebug *debug) {
  TrajectoryPoint target_point;

  if (FLAGS_query_time_nearest_point_only) {
    target_point = trajectory_analyzer.QueryNearestPointByAbsoluteTime(
        Clock::NowInSeconds() + query_relative_time_);
  } else {
    if (FLAGS_use_navigation_mode &&
        !FLAGS_enable_navigation_mode_position_update) {
      target_point = trajectory_analyzer.QueryNearestPointByAbsoluteTime(
          Clock::NowInSeconds() + query_relative_time_);
    } else {
      target_point = trajectory_analyzer.QueryNearestPointByPosition(x, y);
    }
  }

3. 云平台实践教学

3.1 前置步骤

在终端中输入以下指令,启动dreamview

aem bootstrap start

在这里插入图片描述
运行成功后,点击左上角dreamview按钮。当网页中出现apollo后, 即表示dreamview启动成功了, 为了便于调试,在该页面上,我们选择车辆(vehicle) 为MKZ,关闭mute,选择地图Apollo Virutal Map。
在这里插入图片描述
点击Sim Control
在这里插入图片描述
点击Profile, 下载动力学模型,Mkz Model (lincolnmkz动力学模型),Point Mass Model (惯性模型) 。然后选择Dynamic Model,选择Mkz Model.

在这里插入图片描述
启动Planning, Routing, Control,
在这里插入图片描述
在Route Editing页面,选择起点,终点,选择SendRouting Request。发送前点开recorder

在这里插入图片描述
查看车辆前方能够正常生成规划线,此时切换到Tasks页面,点击StartAuto。
在这里插入图片描述
点击Dreamview左侧菜单栏Tasks按钮,关闭LockTask Panel,打开PNC monitor功能按钮,点击右侧Control按钮,可以查看控制相关的数据。
在这里插入图片描述
通过下拉右边栏,可查看控制相关参数的变化,比如右侧第二个图表speed,蓝色曲线表示车辆的实际车速,青绿色的表示规划的参考车速,可看到实际车速与规划速度的跟随情况,从而可以进行控制效果。

3.2 控制参数修改

(1)查看纵向控制参数配置
点击在线编辑,打开控制参数配置文
/apollo_workspace/modules/control/conf/control_conf.pb.txt,可查看apollo默认的控制参数。

找到/apollo/modules/control/conf/control_conf.pb.txt

从配置文件内,可查看横向控制参数如下,横向LQR控制参数主要在lat_controller_conf 内,横向控制参数调节:横向位置误差(q0),横向位置误差变化率(q1),航向误差(q2),航向误差变化率(q3)。

(1)LQR参数

lat_controller_conf {
matrix_q: 0.05
matrix_q: 0.0
matrix_q: 1.0
matrix_q: 0.0
reverse_matrix_q: 0.5
reverse_matrix_q: 0.0
reverse_matrix_q: 0.4
reverse_matrix_q: 0.0
}

(2)横向位置增益,可配置不同速度范围的横向位置调节增益。

  lat_err_gain_scheduler {
    scheduler {
      speed: 4.0
      ratio: 1.0
    }
    scheduler {
      speed: 8.0
      ratio: 0.6
    }
    scheduler {
      speed: 12.0
      ratio: 0.2
    }
    scheduler {
      speed: 20.0
      ratio: 0.1
    }
    scheduler {
      speed: 25.0
      ratio: 0.05
    }
  }

(3)横向航向角增益,可配置不同速度范围的航向角调节增益。

  heading_err_gain_scheduler {
    scheduler {
      speed: 4.0
      ratio: 1.0
    }
    scheduler {
      speed: 8.0
      ratio: 0.6
    }
    scheduler {
      speed: 12.0
      ratio: 0.4
    }
    scheduler {
      speed: 20.0
      ratio: 0.2
    }
    scheduler {
      speed: 25.0
      ratio: 0.1
    }
  }

(4)实验时,使用默认控制参数,进行转向或直行仿真,根据5.2步骤并点击recorder按钮,进行录制数据包。可以通过PNC monitor或control_info工具,对控制效果进行简单分析。
(5)然后将横向位置误差增益和航向角增益系数ratio修改为0,查看转弯画龙效果。

  lat_err_gain_scheduler {
    scheduler {
      speed: 4.0
      ratio: 0.0
    }
    scheduler {
      speed: 8.0
      ratio: 0.0
    }
    scheduler {
      speed: 12.0
      ratio: 0.0
    }
    scheduler {
      speed: 20.0
      ratio: 0.0
    }
    scheduler {
      speed: 25.0
      ratio: 0.0
    }
  }

  heading_err_gain_scheduler {
    scheduler {
      speed: 4.0
      ratio: 0.0
    }
    scheduler {
      speed: 8.0
      ratio: 0.0
    }
    scheduler {
      speed: 12.0
      ratio: 0.0
    }
    scheduler {
      speed: 20.0
      ratio: 0.0
    }
    scheduler {
      speed: 25.0
      ratio: 0.0
    }
  }

打卡notebook,用内置脚本绘图并分析数据。

%matplotlib notebook
run control_info/control_info.py --path /apollo/data/bag/2023-01-19-03-19-59_s/

在这里插入图片描述

未更改时的数据

在这里插入图片描述
更改后的数据,可以看到横向偏差lateral_error最大值达到0.6左右,横摆偏差较大

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/172476.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Android APK 瘦身

Android APK 瘦身的几个方法将项目中的图片由png、jpg转为webp格式。如下操作&#xff1a;1.1选中图片或者含有图片的文件夹 右键选择Convert toWebP..1.2根据自身情况选择有损压缩还是无损压缩备注&#xff1a;官网连接&#xff1a;https://developer.android.google.cn/studi…

leetcode-每日一题-强密码检验器II(简单,数学逻辑)

如果一个密码满足以下所有条件&#xff0c;我们称它是一个 强 密码&#xff1a;它有至少 8 个字符。至少包含 一个小写英文 字母。至少包含 一个大写英文 字母。至少包含 一个数字 。至少包含 一个特殊字符 。特殊字符为&#xff1a;"!#$%^&*()-" 中的一个。它 不…

Golang 从菜鸟到大咖的必经之路_GO 语言的转义字符、注释、规范和代码风格要求

目录 一、GO 语言转义字符 A.Golang 常用的转义字符&#xff08;escape char&#xff09;: B.课程练习 二、Go 语言注释&#xff1a; A.注释&#xff08;Comment&#xff09;: B.Go 语言中的注释类型&#xff1a; C.注释不会被编译 D.shifttab 三、规范的代码风格要求…

聚焦儿童羽绒服产业,看用友YonSuite打造领先实践的数智创新小灯塔

有一种冷“是妈妈觉得你冷”。每每想起小时候&#xff0c;为了应对寒冷的冬季&#xff0c;都会“全副武装”&#xff0c;裹得厚厚的&#xff0c;里三层外三层。 放到如今&#xff0c;有了羽绒服的萌娃们&#xff0c;已不再像我们当年一样穿得厚厚的了。现在的年轻爸妈喜欢装扮…

Udev 为设备节点起别名

查看设备信息&#xff1a; 操作udev&#xff0c;可以使用udevadm命令&#xff0c;如果我们要查看/dev/sda 设备节点信息&#xff0c;我们可以使用下面命令&#xff1a; 命令&#xff1a;udevadm info -a --namesda rootubuntu:/sys/kernel/debug/usb# udevadm info -a --nam…

【C语言】文件操作(1)

文件操作一、关于文件的基础知识1.为什么要学习文件2.什么是文件3.文本文件与二进制文件二、文件的打开和关闭1. 文件指针2.文件的打开和关闭3.文件结尾三、文件的顺序读写1.fputc与fgetc2.fputs与fgets3.fprintf与fscanf4.fwrite与fread5.三个标准流一、关于文件的基础知识 1…

低代码平台的优缺点

全文 1518 字 阅读时间约 5 分钟 本文首发于码匠技术博客 目录 低代码平台的优点 低代码平台的缺点 写在最后 低代码工具大大减少了开发者需要编写的代码量&#xff0c;通过增加可复用代码和组件的数量来帮助企业适应发展需求&#xff0c;设计和部署自定义应用程序以紧跟市…

科研快报|二代加三代扩增子测序探究苏铁植物根部复杂微生物群落组成

背景介绍苏铁俗称铁树&#xff0c;是地球上现存最古老的活化石植物&#xff0c;也是种子植物中最原始的种群。我国是世界上苏铁植物资源最丰富的国家之一&#xff0c;本文对我国的特有种德保苏铁&#xff08;Cycas debaoensis&#xff09;和仙湖苏铁&#xff08;Cycas fairylak…

【复习 自用】JavaScript知识汇总(基础版)

注&#xff1a;之前学过JavaScript&#xff0c;本贴仅用于复习(自用)&#xff0c;建议没基础的朋友先学基础。会混入typescript&#xff01; 更新中~~~~~~~~~~ 对象.属性和对象[‘属性’]的区别&#xff1a;总的来说没啥区别&#xff0c;但是&#xff0c; &#xff01;&#…

办公必备的快捷键,轻松助你键步如飞

作为职场人&#xff0c;你不会只会用Ctrl C和Ctrl V吧&#xff1f;掌握更多的办公快捷键&#xff0c;可以大大提高工作效率&#xff01;今天教大家几个好用的快捷键&#xff01;现在收收藏起来吧&#xff01;快捷键一&#xff1a;winL&#xff08;锁屏&#xff09; 电脑锁屏可以…

【NI Multisim 14.0虚拟仪器设计——放置虚拟仪器仪表(功率表)】

目录 序言 &#x1f34d;放置虚拟仪器仪表 &#x1f349;功率表 序言 NI Multisim最突出的特点之一就是用户界面友好。它可以使电路设计者方便、快捷地使用虚拟元器件和仪器、仪表进行电路设计和仿真。 首先启动NI Multisim 14.0&#xff0c;打开如图所示的启动界面&#x…

opencv的图像基本操作_2

import cv2 #图像BGRimport numpy as npimport matplotlib.pyplot as plt #Matplotlib是RGB腐蚀操作kernel np.ones((3,3), np.uint8) #选择3*3卷积核&#xff0c;3*3括起来&#xff0c;(3,3)erosion cv2.erode(img, kernel, iterations 1) #erode腐蚀函数&#xff0c; iter…

【LeetCode】生命游戏 [M](矩阵)

289. 生命游戏 - 力扣&#xff08;LeetCode&#xff09; 一、题目 根据 百度百科 &#xff0c; 生命游戏 &#xff0c;简称为 生命 &#xff0c;是英国数学家约翰何顿康威在 1970 年发明的细胞自动机。 给定一个包含 m n 个格子的面板&#xff0c;每一个格子都可以看成是一个…

2023年mybatis常见面试题10道

1. MyBatis是什么&#xff1f;Mybatis 是一个半 ORM&#xff08;对象关系映射&#xff09;框架&#xff0c;它内部封装了 JDBC&#xff0c;开发时只需要关注 SQL 语句本身&#xff0c;不需要花费精力去处理加载驱动、创建连接、创建statement 等繁杂的过程。程序员直接编写原生…

Hyper-V三种虚拟网络类型的理解

Windows 10除家庭版之外都内置了Hyper-V&#xff0c;但是Win10系统上Hyper-V虚拟网络的设置方法却与之前版本的Hyper-V有些不同&#xff0c;最近在折腾了无数遍之后&#xff0c;总算有些心得&#xff1a;先说说虚拟网络的三种类型。Hyper-v支持外部、内部、专用三种网络&#x…

蓝桥杯STM32G431RBT6学习——定时器中断

蓝桥杯STM32G431RBT6学习——定时器中断 前言 从网络上的资料来看&#xff0c;蓝桥杯嵌入式好像并不考外部中断&#xff08;毕竟通常只是个按键应用&#xff09;&#xff0c;因此跳过进行定时器的学习。 STM32单片机的定时器通常分为高级定时器、通用定时器、基本定时器三种&…

111、【树与二叉树】leetcode ——669. 修剪二叉搜索树:递归法(C++版本)

题目描述 解题思路 本题的关键是用好递归这个结构&#xff0c;用好每次他向下的遍历和返回的值。每一次递归时&#xff0c;相当于解决与之前相同的问题&#xff0c;因此先按某一种类子问题进行讨论&#xff08;仅有三个结点的满二叉树&#xff09;&#xff0c;当递归的方式向下…

二本学历5年经验的程序员,出去面试被碾压~

目录 1、从一个求职案例引入2 、学历差距&#xff1a;面试官的第一印象3、公司背景差距&#xff1a;你的人生名片4、技术差距&#xff1a;硬核能力的欠缺5、架构能力的差距6、面试结果的分析 这篇文章&#xff0c;聊一个很多人感兴趣的话题&#xff0c;小公司的Java工程师和大…

计网必会:HTPP详解、cookie、缓存

文章目录应用层协议Web和HTTPHTTP 概述采用非持续连接的HTTPRTT 往返时间的定义**三次握手过程**采用持续连接的HTTPHTTP到底采用哪个&#xff1f;HTTP 的报文格式请求报文功效格式响应报文状态码格式Cookie什么是cookieWeb缓存在学习的过程很多人都遇到了HTTP和Cookie&#xf…

【matplotlib】19.基本用法

#【matplotlib】19.基本用法 2021.1.19 画figure图基本方法。参考&#xff1a; https://m.runoob.com/matplotlib/matplotlib-pyplot.html https://mofanpy.com/tutorials/data-manipulation/plt/figure matplotlib 是python的画图包 19.1 基础使用 plot图&#xff0c;就是以…