LOAM学习

news2024/12/22 18:33:17

LOAM

  • Ceres Solver 中的LocalParameterization理解
  • ALOAM雷达里程计主要步骤
    • 论文
    • A-LOAM laser Odometry代码
      • LiDAR Odometry
      • 寻找角点特征
      • 代码流程分析
      • 寻找面点特征
  • 求解器设置

Ceres Solver 中的LocalParameterization理解

该LocalParameterization类用来解决非线性优化中的过参数化问题过参数化指的是:待优化的参数的维度大于其实际意义的维度

例如:旋转需要3个自由度,但是四元数用四个维度表达三个自由度的三维的空间旋转,所以,需要引入单位四元数 限制其中一个自由度。优化问题最重要的就是要计算雅克比矩阵和残差
四元数需要手动自定义加法运算:在ceres中要增加ceres::LocalParameterization来定义加法运算[参考博客请添加图片描述
请添加图片描述
对于增加的参数 旋转矩阵 选择以四元数的方式加入 所以 最终选择4维度 同样 平移向量 选择3维自由度的方式加入
ICP迭代求解最优旋转矩阵与平移向量

ALOAM雷达里程计主要步骤

论文

A 特征提取

选择锐边 和 平面片上的特征点(角点与面点)。(每个激光点云是通过三维激光雷达设备扫描得到的空间点的数据集,每个点云都包含了三维坐标X,Y,Z以及激光反射强度 点云的坐标系:x轴指向汽车的前部,y轴指向汽车的左侧。由于这个坐标系采用右手定则,坐标系z轴指向汽车上方)雷达点云基础知识讲解
根据局部曲面的平滑度(曲率)去筛选特征点(S为激光扫描仪同次扫描返回的i包括一系列连续点的集合,i中左右各有一半数目的点,每两个点之间的间隔为 0.25度,计算完成曲率后,将点按照曲率c进行排序 选择曲率(max)作为角点 选择 曲率(min)作为面点。将每一次扫描分6个子区域==目的:在环境中均匀分布特征点,每个区域最多提供两个边缘点以及4个平面点 (通过for 循环实现)==同时,通过if语句判断 是否i 的 c 值大于或小于阈值且所选点的数量不超过最大值时,才可以将点 i 选择为边缘点或平面点
特征点的选择原则:不选择与激光束平行的点 同时也不选择被遮挡的点作为特征点
特征点选择原则:选择的边缘点或平面点的数量不能超过子区域的最大值。并且它的周围点都没有被选择,并且它不能在大致平行于激光束的平面上,或位于遮挡区域的边界上。
请添加图片描述
公式解释:
请添加图片描述
直观理解:
请添加图片描述如果这11个点均匀分布在同一个直线上,那么求曲率对应的红点附近的点求均值得到的就是点的本身 ,但是如果曲线弯曲了 那么求解完成的点与红点的偏离距离越大证明曲率越大。平面点:在三维空间中处于平滑平面上的点,其和周围点的大小差距不大,曲率较低,平滑度较低。
边缘点:在三维空间中处于尖锐边缘上的点,其和周围点的大小差距较大,曲率较高,平滑度较高。

B. Finding Feature Point Correspondence
里程计odometry在一个扫描帧(sweep)中估计激光雷达的运动。tk为开始扫描的时间,每次扫描结束后,到下一个时刻tk+1开始时刻,在[tk,tk+1]时段感知的点云Pk被通过重投影到tk+1时刻。记为 Pk’ ,在k+1到k+2扫描期间 作为已经实现畸变校正的点云初始值与新接收的点云PK+1一起(未实现畸变校正)估计雷达的运动。
分别在Pk’ 与PK+1中寻找两个激光雷达点云之间的对应关系。对于Pk+1,使用曲率(上一节特征选择算法)从激光雷达点云中查找到边缘点以及平面点由于平面点与边缘点存在多个 所以定义点集Ek+1 和 Hk+1 。然后从Pk’中找出作为Ek+1 中的边缘点的对应,找到平面点作为 Hk+1 中的点的对应。
请添加图片描述> K+1次扫描时候,Pk+1是空集,扫描过程随着接收更多的点而增长,在每次迭代中使用当前估计的变换将 Ek+1 和 Hk+1 重新投影到扫描的开始时刻tk+1。设 Ek+1 ‘和 Hk+1’ 为重投影点集。对于 Ek+1 ‘和Hk+1’ 中的每个点,我们将在Pk’ 中找到最邻近点注意:此时将Ek+1 和 Hk+1 重新投影到扫描的开始时刻tk+1,得到Ek+1 ‘和 Hk+1’,同时Pk’对应的也是tk+1时刻 所以 同时间戳 其相应的特征点能够进行匹配(两次重投影)
边缘线的寻找过程:设 i 是Ek+1’ 中的一个点, i∈Ek+1’ 。边缘线(edge line)由两点表示。定义 j 表示Pk’ 中 i 的最近邻点, j∈Pk’ ,并定义 l 是与 j 相邻的连续两条扫描线束(scan)【上下各一条线束 scan ,相邻两次扫描是为了防止运动畸变过大导致边缘线特征提取不正确】中 i 的最近邻点。 (j,l) 形成 i 的对应关系通过计算曲率验证j和i都是边缘点。这里,我们特别要求 j 和 l 来自不同的扫描线束,因为单个扫描线束(scan)不能包含来自同一边缘线(edge line)的多个点【保证线特征与扫描线不是同一条线】
平面点的平面片的寻找过程: i 为 Hk+1’ 中的一个点, i∈Hk+1’ 。平面片由三个点表示。与上一段类似,我们在 Pk '中找到 i 的最近邻点,表示为 j 。然后,我们找另外两个点, l 和 m 均是 i 的最近邻点,一个在 j 的相同扫描线束中,另一个在 j 的相邻两个连续扫描线束中。这保证了三个点不共线。为了验证 j、l 和 m 都是平面点,我们根据公式 (1) 再次检查局部曲面的平滑度
通过计算从特征点到其对应关系的距离 通过最小化特征点的总距离来恢复激光雷达的运动
请添加图片描述
用到了点到线的距离公式:获取两帧间的数据关联情况,应尽可能的让点到直线的距离最小
请添加图片描述
对于平面点而言:采用点到平面的距离公式:为了获取两帧间的数据关联情况,应尽可能的让点到平面的距离最小(相当于让向量 ij 到向量 jn 的投影的模长最小)
请添加图片描述
理想情况下:边缘点位于边缘线上,平面点位于平面上

C. Motion Estimation (运动状态估计算法)

通过使用非线性最小二乘构建残差函数 优化距离参数 获得具有最优解的旋转矩阵。
激光雷达在扫描的过程中以恒定的角速度和线速度进行运动,这允许我们在扫描中对在不同时间接收到的点进行线性插值计算姿态变换。TL,K+1,i为[tk+1,t]时刻,对应的i点的在雷达坐标系(L系)反映的姿态变换,其中t为当前时间的时间戳。
请添加图片描述
所以对于每一个时刻ti都可以进行插值,得到[tk+1,ti]之间的相对位姿变换。
请添加图片描述
这也就是迭代的意义:相当于假定每一个初始时刻,例如,我们考虑tk-1时刻,经过重投影后,其对应的该时刻开始的初始位姿状态是经过运动补偿的(之前进行线性插值),然后,在当前tk时刻,由于不断的扫描 使得特征点的数量持续增多,每个特征点在时间增加的同时 姿态也在相应发生着变化 所以,通过线性插值方法 可以假定能够用来补偿当前时刻所在点在运动过程中的偏移。上面的插值公式因为TL,K+1是已经进行上次插值补偿的数据 所以 利用上述公式 可以得到小的时间间隔内的姿态变换的运动畸变补偿数据

Ek+1 和 Hk+1 是从 Pk+1 中提取的边缘点集和平面点集,而 Ek+1’和 Hk+1’ 是重新投影到扫描开始的点集。为了得到激光雷达的运动,我们需要在 Ek+1 和 Ek+1’ 或者 Hk+1和 Hk+1’之间建立几何关系:请添加图片描述
也就是:下面这张图像所表示的:
在这里插入图片描述
请添加图片描述请添加图片描述
D. LiDAR Odometry algorithm (里程计算法)

激光雷达里程计算法:将最后一次扫描的点云 Pk '、当前正在扫描的增长点云 Pk+1 和最后一次递归的姿势变换 TLk+1 作为输入【匀速模型】。如果一个新的扫描开始, TLk+1 被设置为零(第 4 - 6 行)。然后,该算法从 PK+1 中提取特征点,在第 7 行构造 Ek+1 和 Hk+1 。对于每个特征点,我们在Pk’中找到其对应关系(第 9 - 19 行)。运动估计适用于鲁棒拟合(robust fitting)。在第 15 行中,算法为每个特征点分配一个二次方权重(bisquare weight)。对与其对应点距离较大的特征点分配较小的权重,对距离大于阈值的特征点视为异常值并分配零权重【类似于核函数】。然后,在第 16 行中,姿势变换迭代更新一次。如果发现收敛或满足最大迭代次数,则非线性优化终止。如果算法到达扫描结束时, Pk+1 将使用扫描期间的运动估计重新投影到时间戳 tk+2 【当前 lidar 直接获得一帧数据,此处判断可以忽略】。否则,下一轮递归只返回转换 TLk+1 (这里并没有表示:在tk+1时刻点云不断增长情况下 对于两个特征数据集的重投影过程,只是说下次递归如果不是新的扫描的话 下一轮递归则返回TLk+1 )。(缺少不断增长的重投影的描述)

A-LOAM laser Odometry代码

值得借鉴的翻译论文

LiDAR Odometry

通过以下的4种类回调函数 将四类特征点分类存储到相应的buffer数据中
请添加图片描述判断是否存在接收数据 同时也必须满足四种角点特征对应的时间戳信息一致
请添加图片描述如果对应有效的特征点数据 我们通过fromROSMsg函数每次将一个feature point 信息传递给对应的pcl::PointCloud::Ptr 数据
请添加图片描述判断系统是否初始化
请添加图片描述

这里需要明确:对于第一次接收到的特征点信息,并不做else对应的重投影以及畸变校正、而是直接执行第二张图片对应的程序块:将接收到的弱角点保存为上一帧弱角点,为下一帧做准备; 把当前帧弱面点保存为上一帧弱面点,为下一帧做准备,然后将对应的弱角点与弱面点 插入到KD树用来后续的几何对应搜索
请添加图片描述

ALOAM
在这里插入图片描述
四元数的插值不能用简单的线性插值,而是用slerp —— 当时间t匀速变化时,代表姿态矢量的角速度变化并不均匀(Eigen::Quaterniond::Identity().slerp(t, q_last_curr)能够实现四元数的球面插值。t ∈ [0, 1]为插值点,q_last_curr为两帧之间的旋转四元数,即针对两帧之间的旋转而线性插入一个四元数。)
系统已经实现初始化
那么在就需要在不断增加角点特征以及面点特征的基础上,执行迭代优化

通过定义corner_correspondence和plane_correspondence判断平面以及角点在不断增加特征点的数量后的对应number数量。
定义局部参数LocalParameterization(LocalParameterization 类的作用是解决非线性优化中的过参数化问题)所谓过参数化,即待优化参数的实际自由度小于参数本身的自由度(四元数本身4个自由度 但是待优化参数实际的自由度是3)——>如果对于四元数进行优化 需要对待优化的四元数参数进行重构——(自定义LocalParameterization的子类实现,因为LocalParaneterization 本身是一个虚基类,用户可以自行定义自己需要使用的子类)
使用ceres::LocalParameterization 将待优化的四元数执行自定义的四元数局部参数化
设置ceres solver的问题的option 以及problem
这里注意:AddParameterBlock函数是显式的增加参数块——因为将待优化四元数进行了参数调整 所以需要显式增加参数信息
这里待优化的变量包括(四元数的旋转以及平移三维向量)请添加图片描述寻找当前角点特征的correspondence对应关系
为优化存在的每一个角点以及面点特征寻找correspondence

寻找角点特征

整体流程:将当前的特征点cornerPointsSharp->points[i]通过重投影TransformToStart函数,得到所研究时段的初始时刻的对应的投影点pointSel,因为通过systeminit已经将弱角点对应的信息全部传递给kdtreeCornerLast存储,所以通过nearestKSearch,能够茶找到pointSel投影点对应的最近邻点信息closestPointInd;然后通过j = closestPointInd + 1以及j = closestPointInd - 1,分别搜索同扇区 不同扇区 不同扫描区的closestPointInd 以及minPointInd2,找到最近邻点后 通过LidarEdgeFactor::Create 创建自动求导函数new ceres::AutoDiffCostFunction<LidarEdgeFactor, 3, 4, 3>并加入problem.AddResidualBlock(cost_function, loss_function, para_q, para_t);残差函数

代码流程分析

TransformToStart(&(cornerPointsSharp->points[i]), &pointSel);

判断是否存在畸变运动补偿DISTORTION,而对于点云的畸变运动补偿的处理方法此处选择:通过四元数slerp插值的方法实现
因为假定为匀速运动 所以每一帧之间的相对姿态变换都是相同的 所以通过调用Quaterniond.slerp函数 能够实现相对时刻的插值(插值得到当前点相对于该帧起始点的旋转)
Eigen::Quaterniond q_point_last = Eigen::Quaterniond::Identity().slerp(s, q_last_curr);
然后这里需要注意:Eigen::Vector3d un_point = q_point_last * point + t_point_last;是将重投影的关键(将相对时间中的任何时刻都统一到开始时刻 也就相当于去除了畸变 完成了运动补偿)
请添加图片描述
重投影到每一个相对变换的时间间隔的初始时刻start time,通过nearestKSearch函数搜索pointSel(当前初始时刻投影点对应的最近邻) 也就是论文中的j点
因为是边缘线 ,在已经校正的点云上面找到对应的最近邻角点信息
(因为之前论文阐述的角点筛选原则:不能在同scan以及不是在连续的相邻scan中 那么分别执行
if (int(laserCloudCornerLast->points[j].intensity) <= closestPointScanID)// if in the same scan line, continue continue; if (int(laserCloudCornerLast->points[j].intensity) > (closestPointScanID + NEARBY_SCAN))// if not in nearby scans, end the loop break;
如果找到对应的最近邻点来自相邻射线计算相邻的short distance (pointSqDis)
j = closestPointInd + 1在下方查找j点的最近邻l点 请添加图片描述j = closestPointInd - 1 在上方查找最邻近点,查找j的最近邻l点
如果 当前j的最近邻点与j同scan那么 continue
如果 非 consecutive scan那么break
如果满足consecutive scan 同时满足最近邻点筛选的条件 那么计算pointSqDis,并且通过:
if (pointSqDis < minPointSqDis2) { // find nearer point minPointSqDis2 = pointSqDis; minPointInd2 = j; }迭代选择最终的最近邻点
请添加图片描述构建损失函数 为问题增加残差块
请添加图片描述> 同理 对于边缘线特征来说,如果投影点位于最近邻点所在的线上,那么,residual就为0,但是 由于误差的存在 不可能使得residual为0 所以 出现edge factor因子。请添加图片描述
LidarEdgeFactor::Create函数构建自动求导构建自动求导的损失函数 这个损失函数 第一维数据为residual3维,第二个为经过localparameterazation的4维四元数数据 第三位为平移向量的3维度数据。创建LidarEdgeFactor因子请添加图片描述> 根据距离公式计算残差residual
这里首先将当前位置的current_pose信息经过四元数插值转换到所研究时段的初始时刻。然后,根据点到线的距离公式 计算残差 这里将点到直线的距离分别分配到x,y,z三个方向上的residual

寻找面点特征

前面的重投影以及在重投影集中寻找i点对应的最近邻j点是相同的步骤
请添加图片描述> 基于搜索到的最近邻点j,通过两个for循环实现向前查找以及向后查找(注意这里laserCloudSurfLast->points是根据曲率进行排序得到的,如果不是nearest scan 直接break)两个for 循环都是用来寻找consecutive scan以及same scan的,所以当int(laserCloudSurfLast->points[j].intensity) >= closestPointScanID 或者<=closestPointScanID 时候 改变的都是minPointInd2 以及minPointSqDis2 代表same scan存在(l点) 同理int(laserCloudSurfLast->points[j].intensity) > closestPointScanID或者 < closestPointScanID代表consecutive scan的邻近点 最终选择minPointInd3 以及以及minPointSqDis2存储(m点)
closestPointInd对应论文中的j点、minPointInd2对应论文中的、minPointInd3对应论文中的。

请添加图片描述搜索到三个非共线平面点后 构建损失函数
请添加图片描述基于LidarPlaneFactor的因子
利用点到平面的距离公式 得出一维残差,其实可以这么理解:求点到平面的距离(如果是平面点的话,那么最近邻点和搜索点共面 那么残差就为0了 但是因为误差的存在 不可能残差为0 所以 计算点积)
请添加图片描述

求解器设置

请添加图片描述
这里的t_last_curr与q_last_curr通过ceres solver 不断优化para_q以及para_t来

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

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

相关文章

最全的软件测试面试题(含答案)

软件的生命周期(prdctrm) 计划阶段(planning)-〉需求分析(requirement)-〉设计阶段(design)-〉编码(coding)->测试(testing)->运行与维护(running maintrnacne) 测试用例 用例编号  测试项目  测试标题  重要级别  预置条件  输入数据  执行步骤   预期结果 1…

python做游戏好用吗

Python做游戏是完全可以的&#xff0c;而且也非常简单&#xff0c;有一个专门针对游戏开发的平台&#xff08;模块&#xff09;—pygame&#xff0c;允许开发人员快速设计游戏而又摆脱了低级语言的束缚&#xff0c;下面我简单介绍一下这个模块的安装和使用&#xff1a; 1、首先…

Java手写RPC框架-01-开篇

项目背景 随着业务不断升级&#xff0c;系统规模不断扩大&#xff0c; 单体架构会产生越来越多的问题&#xff0c;需要引入微服务将原先架构解耦为一个个模块。每个服务模块放在不同的服务器上&#xff0c;能够保证系统在高并发环境下的正常运转。 各个服务模块之间如何相互调…

想了解医疗大模型吗?请看《智能系统学报》实验室最新综述论文

本文改编自实验室的最新综述论文《医疗领域的大型语言模型综述》&#xff0c;该论文发表于《智能系统学报》。《智能系统学报》是中国人工智能学会会刊、“中国人工智能学会推荐中文学术期刊”列表中的A类期刊。该论文合作单位包括上海理工大学、上海儿童医学中心、复旦大学附属…

LangChain-Chatchat本地搭建部署

文章目录 前言一、安装部署1.软硬件要求2. 安装 Langchain-Chatchat3.安装Xinference4.遇到的问题问题1&#xff1a;Failed building wheel for llama-cpp-python问题2&#xff1a;Failed building wheel for pynini问题3&#xff1a;运行xinference错误 二、初始化项目配置并运…

了解软件测试的概念

本文我们来了解软件测试 的一些基本概念。同时需要记住衡量软件测试结果的依据—需求&#xff1b; 1. 需求的概念 满足用户期望或正式规定文档&#xff08;合同、标准、规范&#xff09;所具有的条件和权能&#xff0c;包含用户需求和软件需求。&#xff08;其实就是客户想要软…

摩尔信使MThings逻辑控制实例——交通灯

摩尔信使MThings提供了强大的数据配置和逻辑控制功能&#xff0c;可为用户带来一种高效且直观的方式进行管理和控制交通灯系统。与传统的PLC&#xff08;可编程逻辑控制器&#xff09;相比&#xff0c;MThings的界面更加用户友好&#xff0c;使得即使是非专业的用户也能够轻松地…

在 Mac 中设置环境变量

目录 什么是环境变量&#xff0c;为什么它们重要&#xff1f;什么是环境变量&#xff1f;举个例子 如何查看环境变量如何设置和修改环境变量1. 临时设置环境变量2. 永久设置环境变量3. 修改现有环境变量 环境变量在开发中的应用在 Node.js 项目中使用环境变量在 Python 项目中使…

Certificate has expired(npm 安装strapi)

报错信息 解决方法 1、清空缓存&#xff0c;有时&#xff0c;损坏的缓存会导致连接问题 npm cache clean --force 2、切换到淘宝镜像源的 npm 注册表 npm config set registry https://registry.npmmirror.com/ 执行这两步后就可以执行自己想要安装的东西了&#xff0c;我是在执…

Uniapp + Vue3 + Vite +Uview + Pinia 实现购物车功能(最新附源码保姆级)

Uniapp Vue3 Vite Uview Pinia 实现购物车功能&#xff08;最新附源码保姆级&#xff09; 1、效果展示2、安装 Pinia 和 Uview3、配置 Pinia4、页面展示 1、效果展示 2、安装 Pinia 和 Uview 官网 https://pinia.vuejs.org/zh/getting-started.html安装命令 cnpm install pi…

云轴科技ZStack 获鲲鹏应用创新大赛2024上海赛区决赛一等奖

9月13日&#xff0c;鲲鹏应用创新大赛2024上海赛区决赛成功举办。经评委专家从方案创新性、技术领先性、商业前景以及社会价值四个维度严格评审&#xff0c;云轴科技ZStack参赛作品《ZStack鲲鹏原生开发方案》荣获上海赛区企业赛——原生开发赛道&#xff08;互联网&#xff09…

AI大模型系统实战:挑战与应用多领域,人工智能大模型的实际应用场景

AI大模型系统实战&#xff1a;挑战与应用多领域&#xff0c;人工智能大模型的实际应用场景 人工智能的新浪潮中&#xff0c;大模型系统已成为技术革新的重要驱动力。它们以其强大的学习能力和广泛的应用场景&#xff0c;正在重新定义我们与机器交互的方式。本文将深入探讨AI大模…

VS 如何显示构建的时间

Cherno 构建 Hazel 的时候会显示构建时间 VS -> Tools -> Options -> Projects and Solution -> VC Project Settings

UGit:腾讯自研的Git客户端新宠

UGit 是一款专门针对腾讯内部研发环境特点量身定制的 Git 客户端&#xff0c;其目标在于大幅提升开发效率以及确保团队协作的高度流畅性。UGit 能够良好地支持 macOS 10.11 及以上版本、Apple Silicon 以及 Win64 位系统。 可以下载体验一把。 https://ugit.qq.com/zh/index.…

GIS可视化软件:地理信息与遥感领域中的洞察之眼

在地理信息与遥感技术的广阔天地中&#xff0c;可视化软件如同一双洞察世界的明眸&#xff0c;将复杂的数据编织成生动、直观的画卷&#xff0c;为我们揭示地球的奥秘与城市的律动。本文将深入挖掘其技术核心、应用实例、未来趋势&#xff0c;探讨可视化软件如何为地理信息与遥…

电能质量监测装置和防孤岛装置在特斯拉工厂分布式光伏项目的应用

摘要&#xff1a;全球对可再生能源的关注增加&#xff0c;推动了分布式光伏发电系统的普及。这些系统因环保和灵活性而受到青睐&#xff0c;有助于解决能源和环境问题。电能质量在线监测装置和防孤岛保护装置在这些项目中至关重要。监测装置实时跟踪电压和电流&#xff0c;保障…

切换淘宝最新镜像源npm详细讲解

​ 大家好&#xff0c;我是程序员小羊&#xff01; 前言&#xff1a; 在中国大陆&#xff0c;npm&#xff08;Node Package Manager&#xff09;的默认源由于网络限制&#xff0c;速度可能较慢。为了解决这个问题&#xff0c;淘宝提供了一个镜像源&#xff0c;它同步了 npm 的…

连接数据库(以MySQL为例)

文章目录 前言一、数据库是什么&#xff1f;二、连接步骤 1.手动导入驱动包2.连接数据库总结 前言 面对应用程序的开发&#xff0c;普遍需要保存用户的海量数据。保存粮的库叫粮库&#xff0c;保存水的库叫水库&#xff0c;那么保存数据的库自然叫数据库。有了数据库&#xff0…

Vue的slot插槽(默认插槽、具名插槽、作用域插槽)

目录 1. slot插槽1.1 默认插槽1.2 具名插槽1.3 作用域插槽 1. slot插槽 作用&#xff1a;让父组件可以向子组件指定位置插入html结构&#xff0c;也是一种组件间通信的方式&#xff0c;适用于父组件向子组件传递数据 1.1 默认插槽 Category.vue&#xff1a; 定义一个插槽。…

先楫HPM6750 Windows下VSCode开发环境配置

用的是EVKmini&#xff0c;ft2232作为调试器jtag接口调试 启动start_gui.exe 以hello_world为例&#xff0c;更改一下build path&#xff0c;可以generate并使用gcc compile 最后会得到这些 点击start_gui里面的命令行&#xff0c;用命令行启动vscode 新建.vscode文件夹&…