一起自学SLAM算法:8.2 Cartographer算法

news2025/1/12 16:13:18

连载文章,长期更新,欢迎关注:

Gmapping代码实现相对简洁,非常适合初学者入门学习。但是Gmapping属于基于滤波方法的SLAM系统,明显的缺点是无法构建大规模的地图,这一点已经在第7章中讨论过了。而基于优化的方法则可以构建大规模的地图,基于优化的方法实现的激光SLAM算法也有很多,比如Hector、Karto、Cartographer等。而Cartographer是其中获好评最多的算法,Cartographer是这几种算法中提出时间最新、开发团队来自著名的谷歌公司、代码的工程稳定性较高、少有的建图和重定位兼具的算法。所以,下面将从原理分析、源码解读和安装与运行这3个方面展开讲解Cartographer算法。

8.2.1 Cartographer原理分析

其实基于优化方法的激光SLAM已经不是一个新研究领域了,谷歌的Cartographer算法主要是在提高建图精度和提高后端优化效率方面做了创新。当然Cartographer算法在工程应用上的创新也很有价值,Cartographer算法最初是为谷歌的背包设计的建图算法,谷歌的背包是一个搭载了水平单线激光雷达、垂直单线激光雷达和IMU的装置,用户只要背上背包行走就能将环境地图扫描出来。由于背包是背在人身上的,最开始Cartographer算法是只支持激光雷达和IMU建图的,后来为了适应移动机器人的需求,将轮式里程计、GPS、环境已知信标也加入到算法,也就是说Cartographer算法是一个支持多激光雷达、IMU、轮式里程计、GPS、环境已知信标的传感器融合建图算法。下面将结合Cartographer算法的核心论文[3]对Cartographer的原理展开分析。

基于优化方法的SLAM系统通常采用前端局部建图、闭环检测和后端全局优化这种经典框架,如图8-8所示。

图8-8  基于优化的SLAM经典框架

1.局部建图

局部建图就是利用传感器扫描数据构建局部地图的过程,在第7章中已经介绍过,机器人位姿点、观测数据和地图之间通过约束量建立联系。如果在机器人位姿准确的情况下,可以把观测到的路标直接添加进地图。由于从机器人运动预测模型得到的机器人位姿存在误差,所以需要先用观测数据对这个预测位姿做进一步更新,以更新后的机器人位姿为基准来将对应的观测加入地图。用观测数据对这个预测位姿做进一步更新,主要有下面几种方法:

  • Scan-to-scan matching
  • Scan-to-map matching
  • Pixel-accurate scan matching

最简单的更新方法,就是Scan-to-scan matching方法。由于机器人相邻两个位姿对应的雷达扫描轮廓存在较大的关联性,在预测位姿附近范围内将当前帧雷达数据与前一帧雷达数据进行匹配,以匹配位姿为机器人位姿的更新量。但是,单帧雷达数据包含的信息太少了,直接拿相邻两帧帧雷达数据做匹配更新会引入较大误差,并且雷达数据更新很快,这将导致机器人位姿的误差快速累积。

而Scan-to-map matching方法则不同,其采用当前帧雷达数据与已构建出的地图做匹配。由于已构建出的地图信息量相对丰富稳定,所以并不会导致机器人位姿的误差累积过快的问题,如图8-9所示。Cartographer的局部建图就是采用这种方法,也称为局部优化。

而Pixel-accurate scan matching方法,其匹配窗口内的搜索粒度更精细,这样能得到精度更高的位姿,缺点是计算代价太大,后面将讲到的Cartographer闭环检测采用的就是这种方法。当然,关于闭环检测也有很多别的方法,比如extracted features matching方法、histogram-based matching方法、machine learning方法等,感兴趣的读者可以查阅相关资料。

图8-9  scan-to-scan与scan-to-map对比

在介绍Cartographer局部建图的具体过程之前,需要先了解一下Cartographer地图的组成形式。Cartographer采用局部子图(submap)来组织整个地图,其中若干个激光雷达扫描帧(scan)构成一个submap,然后所有的submap构成全局地图(submaps),如图8-10所示。

图8-10  Cartographer地图结构

不管是雷达扫描帧(scan),局部子图(submap),还是全局地图(submaps),它们之间都是通过位姿关系进行关联的。这里只讨论2D SLAM建图,所以位姿坐标可以表示为\xi =(\xi _{x},\xi _{y},\xi _{\theta })。假设机器人初始位姿为\xi _{1}=(0,0,0),该位姿处雷达扫描帧为scan(1),并利用scan(1)初始化第一个局部子图submap(1)。利用Scan-to-map matching方法计算scan(2)相应的机器人位姿\xi _{2},并基于位姿\xi _{2}将scan(2)加入submap(1)。不断执行Scan-to-map matching方法添加新得到的雷达帧,直到新出现的雷达帧完全包含在submap(1)中时,换句话说就是新雷达帧观测不到submap(1)之外的新信息时,就结束submap(1)的创建。这里假设submap(1)由scan(1)、scan(2)和scan(3)构建而成,然后重复上面的步骤构建新的局部子图submap(2)。而所有局部子图{submap(m)}就构成了最终的全局地图submaps。

可以发现,每个雷达扫描帧都对应一个全局地图坐标系下的全局坐标,同时该雷达扫描帧也被包含在对应的局部子图中,也就是说该雷达扫描帧也对应一个局部子图坐标系下的局部坐标。而每个局部子图以第一个插入的雷达扫描帧为起始,该起始雷达扫描帧的全局坐标也就是该局部子图的全局坐标。这样的话,所有雷达扫描帧对应的机器人全局位姿\Xi ^{s}=\left \{ \xi _{j}^{s} \right \},j=1,2,...,n和所有局部子图对应的全局位姿\Xi ^{m}=\left \{ \xi _{i}^{m} \right \},i=1,2,...,m通过Scan-to-map matching产生的局部位姿\xi _{ij}进行关联,这些约束实际上就构成了位姿图。当检测到闭环时,对整个位姿图中的所有位姿量进行全局优化,那么\Xi ^{s}\Xi ^{m}中的所有位姿量都会得到修正,每个位姿上对应的地图点也相应的得到修正,这就是全局建图。接下来详细介绍一下利用Scan-to-map matching方法构建局部子图的过程。

局部子图构建过程涉及到很多坐标系变换的内容,这里就来详细讨论。首先雷达扫描一圈得到的距离点\left \{ h_{k} \right \},k=1,2,...,K是基于雷达旋转中心为坐标系的取值。那么在一个局部子图中,以第一帧雷达位姿为参考,后加入的雷达帧位姿用相对转移矩阵T_{\xi }=(R_{\xi },t_{\xi })表示。这样的话,雷达帧中的数据点h_{k}就可以用式(8-10)所示的公式转换成局部子图坐标系中表示。

与Gmapping类似,Cartographer中的子图(submap)也采用概率栅格地图(Probability Grids)。所谓概率栅格地图,就是连续2D空间被换分成一个个离散的栅格,栅格的边长r为分辨率,通常栅格地图的分辨率r=5cm。那么扫描到的障碍点就替换成用该障碍点所占据的栅格表示。用概率来描述栅格中是否有障碍物,概率值越大说明存在障碍物的可能性越高。

接下来就可以讨论新雷达数据被加入到子图(submap)的过程了,先按式(8-10)将新雷达数据转换到子图坐标系,这时候新雷达数据点会覆盖子图的一些栅格\left \{ M_{old} \right \},每个栅格存在3种状态,即未知(unknown)、非占据(miss)和占据(hit)。如图8-11所示,雷达扫描点所覆盖的栅格就应该为占据(hit)状态;而雷达扫描光束起点与终点区域内肯定就没有障碍物,该区域覆盖的栅格就应该为非占据(miss)状态;由于雷达扫描分辨率和量程限制,未被雷达扫描点所覆盖的栅格就应该为未知(unknown)状态。由于子图(submap)中的栅格可能不只被一帧雷达扫描点所覆盖,所以需要对栅格的状态进行迭代更新,具体分下面2种情况处理。

图8-11  雷达数据点栅格化

情况1,在当前帧新雷达数据点覆盖的栅格\left \{ M_{old} \right \}中,如果该栅格之前从未被雷达数据点覆盖,即未知状态,那么直接用式(8-11)执行初始更新。其中,栅格x若是被新雷达数据点标记为占据(hit)状态,那么就用占据概率P_{hit}给该栅格赋予初值;同理栅格x若是被新雷达数据点标记为非占据(miss)状态,那么就用非占据概率P_{miss}给该栅格赋予初值。概率P_{hit}P_{miss}的取值有雷达概率观测模型给出,见第7章相关内容。

情况2,在当前帧新雷达数据点覆盖的栅格\left \{ M_{old} \right \}中,如果该栅格之前已经被雷达数据点覆盖过,也就是栅格已经有取值M_{old},那么就用式(8-12)执行迭代更新。其中,栅格x若是被新雷达数据点标记为占据(hit)状态,那么就用占据概率P_{hit}M_{old}进行更新;同理栅格x若是被新雷达数据点标记为非占据(miss)状态,那么就用非占据概率P_{miss}M_{old}进行更新。式中odds是一个反比例函数,odds^{-1}odds的反函数。而clamp是一个区间限定函数,当函数值超过设定区间的最大值时都取最大值处理,当函数值超过设定区间的最小值时都取最小值处理。

Cartographer所采用的这种栅格更新机制,能有效降低环境中动态障碍物的干扰。比如建图过程中出现了一个行走的人,那么行走的人经雷达扫描后出现在局部子图栅格的位置每次都不同。假如前一时刻栅格x上出现人,M_{old}被占据概率P_{hit}赋予了初值;而下一时刻,由于人的位置移动了,M_{old}此时将被标记成非占据(miss)状态,由式(8-11)所示更新方法可知,用非占据概率P_{miss}M_{old}进行更新后,栅格x的概率取值变小了。随着越来越多次更新,栅格x的概率将接近0,也就是说动态障碍物被清除掉了。

 以上所讨论的新雷达数据加入到子图(submap)的操作,基于雷达位姿\xi误差较小的条件。由于从机器人运动预测模型得到的机器人位姿存在较大误差,所以需要先用观测数据对这个预测位姿做进一步更新,以更新后的机器人位姿为基准来将对应的观测加入地图。Cartographer中采用了所谓的Scan-to-scan matching方法,对雷达位姿\xi做局部优化,下面讨论具体过程。

在将新雷达数据加入到子图(submap)之前,先在运动预测出的雷达位姿附近窗口内做搜索匹配,如式(8-13)所示。其实这就是一个非线性最小二乘问题,Cartographer采用了自家的Ceres非线性优化工具来求解该问题,关于Ceres的内容已经在第7章介绍过了。而式中的约束量由函数M_{smooth}构建,M_{smooth}是一个双立方插值(bicubic interpolation),也叫平滑。M_{smooth}其实就是用来确定雷达扫描轮廓T_{\xi }\bullet h_{k}与局部子图(submap)之间的匹配度,匹配度取值范围为[0,1]区间。

2.闭环检测

上面介绍的局部建图过程,采用了式(8-12)对位姿\xi进行了局部优化,有效降低了局部建图中的累积误差。但是随着建图规模的扩大,比如上千平米的地图时,总的累积误差还是会很大,也就是在地图构建得很大时,地图出现重影的现象,如图8-12所示。

图8-12  地图重影

其实就是机器人在运动了很远的距离又回到了之前走过的地方,由于局部建图位姿累积误差的存在,使得当前机器人位姿与之前走过的同一个地方并不重合,也就是说真实情况里,这两个地方应该是同一个。自然这两个机器人位姿对应的局部地图也就不重合了,就出现了所谓的重影。借助闭环检测技术,可以检测到机器人位姿闭环这一情况,将闭环约束加入整个建图约束中,并对全局位姿约束进行一次全局优化,这样就能得出全局建图结果。下面将主要对闭环检测过程过程进行讲解。

在局部建图过程中,使用了Scan-to-scan matching方法进行位姿\xi局部优化。而闭环检测中,搜索匹配的窗口W更大,位姿\xi计算精度要求更高,所以需要采用计算效率更高精度更高的搜索匹配算法。首先来看一下回环检测问题的数学表达,如式(8-14)所示。式中的M_{nearest}函数值其实就是雷达数据点T_{\xi }\bullet h_{k}覆盖的栅格所对应概率取值,当搜索结果\xi就是当前帧雷达位姿真实位姿时,当前帧雷达轮廓与地图匹配度很高,即每个M_{nearest}函数值都较大,那么整个求和结果也就最大。

针对式(8-14)所示求最值的问题,最简单的方法就是在窗口W内暴力搜索。假设所选窗口W大小为10m\times 10m,搜索步长为\Delta x=\Delta y=1cm,同时方向角搜索范围假设为30°,搜索步长为\Delta \theta =1^{\circ},那么总的搜索步数为10^{3}\times 10^{3}\times 30=3\times 10^{7}步。每步搜索都要计算式(8-14)所示的匹配得分,如式(8-15)所示。

可以发现,每步搜索都要计算K 维数据的求和运算,而整个搜索过程的计算量为10^{3}\times 10^{3}\times 30 \times K=3\times 10^{7}\times K。虽然暴力搜索匹配可以避免陷入局部最值的问题,但是计算量太大根本没法在机器人中做到实时计算。这种暴力搜索,就是所谓的Pixel-accurate scan matching方法。

采用暴力搜索来做闭环检测显然行不通,因此谷歌在Cartographer中采用分支定界(branch-and-bound)方法来提高闭环检测过程的搜索匹配效率。分支定界简单点理解,就是先以低分辨率的地图来做匹配,然后逐步提高分辨率,如图8-13所示。假设地图原始分辨率为r=1cm,将其进行平滑模糊处理得到分辨率为r=2cm的地图,继续平滑模糊处理可以得到分辨率为r=4cm和r=8cm的地图。

图8-13  地图分辨率

现在来考虑在不同分辨率地图,其搜索窗口W的策略。为了讨论方便,这里举一个简单例子,假设所选取的窗口W=16cm\times 16cm。先以r=8cm分辨率最低的地图开始搜索,这时候窗口W可以按照分辨率被化分成4个区域,也就是4个可能的解,用式(8-15)计算每个区域的匹配得分。选出得分最高的那个区域,将该区域作为新的搜索窗口,并在r=4cm分辨率的地图上开始搜索,同样将窗口按照分辨率化分成4个区域,用式(8-15)计算每个区域的匹配得分。不断重复上面的细分搜索过程,直到搜索分辨率达到最高分辨率为止。整个细分搜索过程,如图8-14所示。

图8-14  细分搜索过程

上面介绍的这种分支定界策略属于广度优先搜索,就是先横向比较同一分辨率下的划分区域的匹配得分,找到得分最高的区域继续划分。而Cartographer中用到的分支定界策略是深度优先搜索,也就纵向比较不同分辨率下划分区域的匹配得分。关于广度优先搜索和深度优先搜索的过程,如图8-15所示。当然在Cartographer分支定界深度优先搜索中,搜索树并不是简单的二叉树,可能每个父节点会分出多个子节点。父节点比子节点代表解空间的分辨率要低,搜索遍历过程会不断进行匹配得分界限判断,并将不符合条件的分支节点进行剪支,大大缩小了搜索空间的维度。通过分支定界法搜索匹配得到的位姿\xi还可以利用前面的Scan-to-map matching方法进一步提高精度,就不多说了。

图8-15  广度优先搜索与深度优先搜索

3.全局建图

闭环检测是在程序后台持续运行的,传感器每输入一帧雷达数据,都要对其进行闭环检测。当闭环检测中匹配得分超过设定阈值就判定闭环,此时将闭环约束加入整个建图约束中,并对全局位姿约束进行一次全局优化,这样就能得出全局建图结果,下面详细介绍全局优化过程。

在Cartographer中采用的是稀疏位姿图来做全局优化,稀疏位姿图的约束关系可以从图8-10中构建。所有雷达扫描帧对应的机器人全局位姿\Xi ^{s}=\left \{ \xi _{j}^{s} \right \},j=1,2,...,n和所有局部子图对应的全局位姿\Xi ^{m}=\left \{ \xi _{i}^{m} \right \},i=1,2,...,m通过Scan-to-map matching产生的局部位姿\xi _{ij}进行关联,数学表达如式(8-16)所示。

 式中,所有雷达扫描帧对应的机器人全局位姿用集合\Xi ^{s}=\left \{ \xi _{j}^{s} \right \},j=1,2,...,n表示,j是雷达扫描帧的序号;所有局部子图对应的全局位姿用集合\Xi ^{m}=\left \{ \xi _{i}^{m} \right \},i=1,2,...,m,i是子图的序号。而雷达扫描数据在局部子图中还具有局部位姿,比如\xi _{ij}表示序号为j的雷达扫描帧在序号为i的局部子图中的局部位姿,该局部位姿通过Scan-to-map matching方法确定。而损失函数\rho用于惩罚那些过大的误差项,比如Huber损失函数。可以看出式(8-16)所示的问题其实这就是一个非线性最小二乘问题,Cartographer同样采用了自家的Ceres非线性优化工具来求解该问题。

当检测到闭环时,对整个位姿图中的所有位姿量进行全局优化,那么\Xi ^{s}\Xi ^{m}中的所有位姿量都会得到修正,每个位姿上对应的地图点也相应的得到修正,这就是全局建图。

以上分析只是站在Cartographer原理的角度展开讨论的,也就是说Cartographer算法具体源码实现过程的细节可能会与上面的分析有所出入,读者请以源码为准。

8.2.2 Cartographer源码解读

上面讨论完Cartographer的原理,现在就来解读Cartographer的源码,其代码框架如图8-16所示。可以看出Cartographer算法主要由3部分组成,分别为cartographer_ros功能包、cartographer核心库和ceres-solver非线性优化库。本书写作时,最稳定的版本为cartographer_ros-1.0.0+cartographer-1.0.0+ceres-solver-1.13.0,所以下面的分析将以此版本代码展开。

图8-16  Cartographer代码框架

1.cartographer_ros功能包

(先占个坑,有时间再来补充详细内容,大家可以直接看文后的参考文献)

2.cartographer核心库

(先占个坑,有时间再来补充详细内容,大家可以直接看文后的参考文献)

3.ceres-solver非线性优化库

(先占个坑,有时间再来补充详细内容,大家可以直接看文后的参考文献)

8.2.3 Cartographer安装与运行

学习完Cartographer算法的原理及源码之后,大家肯定迫不及待想亲自安装运行一下Cartographer体验一下真实效果。在第1章中已经声明过,本书在Ubuntu18.04和ROS melodic环境下进行讨论。不管是使用X86主机、X86主机虚拟机还是ARM主机,一旦装好Ubuntu18.04系统后,就可以在该系统上安装ROS melodic发行版了。如果你只是想利用数据集离线跑算法,可以选择在X86主机或X86主机虚拟机上运行Ubuntu18.04和ROS melodic,关于这一部分的环境搭建请参考1.2.1节的内容。如果需要在实际机器人上在线跑算法,可以选择在ARM主机上运行Ubuntu18.04和ROS melodic,关于这一部分的环境搭建请参考第5章的内容。所以,下面的讨论假设Ubuntu18.04和ROS melodic环境已经准备妥当了。下面所讨论的Cartographer安装、配置和运行的内容都参考自cartographer_ros官方文档和cartographer官方文档。

1.Cartographer安装

(先占个坑,有时间再来补充详细内容,大家可以直接看文后的参考文献)

2.Cartographer在实际机器人运行

(先占个坑,有时间再来补充详细内容,大家可以直接看文后的参考文献)

参考文献

【1】 张虎,机器人SLAM导航核心技术与实战[M]. 机械工业出版社,2022.

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

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

相关文章

Python爬虫之findall和lxml

Python爬虫之findall和lxml 提示:前言 Python爬虫之findall和lxml 提示:写完文章后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录Python爬虫之findall和lxml前言一、导入包二、设置URL和获取视频链接三、解析视频名字…

31. 实战:PyQuery获取小电视Top100详细信息(文末源码)

目录 前言 (链接放在评论区)(链接放在评论区)(链接放在评论区) 目的 (链接放在评论区)(链接放在评论区)(链接放…

趣味三角——第4章——三角学迈向解析化

第4章 三角学迈向解析化(或分析化) 目录 4.1 三角学迈向解析化的过程简述 4.2 Franois Vieter对三角学解析化的贡献 “Thus the analysis of angular sections involves geometric and arithmetic secrets which hitherto have been penetrated by no one(因此&#xf…

Idea中指定xml文件失效

目录一、🐇 项目场景:二、🐇 问题描述三、🐇 原因分析:四、🐇 解决方案:一、🐇 项目场景: 最近狮子在搞一个项目,需要用到数据库多表查询,所以在…

数据挖掘,计算机网络、操作系统刷题笔记35

数据挖掘,计算机网络、操作系统刷题笔记35 2022找工作是学历、能力和运气的超强结合体,遇到寒冬,大厂不招人,可能很多算法学生都得去找开发,测开 测开的话,你就得学数据库,sql,orac…

【论文翻译】Jointformer :一种基于误差预测和改进的三维人体姿态估计的单帧提升变压器

摘要 单目三维人体姿态估计技术有望极大地提高人体运动数据的可用性。表现最好的单幅图像2D3D提升模型使用图卷积网络(GCNs),通常需要一些手动输入来定义不同身体关节之间的关系。我们提出了一种新的基于变压器的方法,该方法使用更广泛的自我注意机制来…

nodejs+vue高校网上报名系统

本课题利用nodejsVue设计实现网上报名系统。系统的主要功能是:用户在线注册信息之后,利用注册时填写的用户账号与密码,登入系统后,对注册的个人信息进行修改,在线报名,能正确的提交有送报考的基本信息&…

【图卷积网络】01-卷积神经网络:从欧氏空间到非欧氏空间

人工神经网络发展浪潮 第三次浪潮——卷积神经网络 加拿大多伦多大学教授,机器学习领域泰斗Geoffery Hinton及其学生在《科学》上发表了一篇论文 (Hinton, G. E . Reducing the Dimensionality of Data with Neural Networks[J]. Science, 2006, 313(578…

【Typescript学习】使用 React 和 TypeScript 构建web应用(二)部分UI、useState、useRef、Props

教程来自freecodeCamp:【英字】使用 React 和 TypeScript 构建应用程序 跟做,仅记录用 其他资料:https://www.freecodecamp.org/chinese/news/learn-typescript-beginners-guide/ 第二天 以下是视频(0:18-0:40) 的内容 目录第二天1 App 函数…

【二叉树】java实现代码,详解二叉树,带大家更深刻的掌握二叉树递归思想

前言: 大家好,我是良辰丫🪐🪐🪐,在探索数据结构的旅程中,二叉树可以说是数据结构中的重点,笔试面试经常出现的问题,同时也是难点。🐥🐥&#x1f4…

【Java开发】Spring Cloud 09 :微服务网关 Gateway

Spring Cloud Gateway(简称 Gateway),它在微服务架构中扮演的角色是“微服务网关”,Nginx 和 Gateway 在微服务体系中的分工是不一样的。Gateway 作为更底层的微服务网关,通常是作为外部 Nginx 网关和内部微服务系统之…

Markdown编辑器基本语法

这里写自定义目录标题欢迎使用Markdown编辑器新的改变功能快捷键合理的创建标题,有助于目录的生成如何改变文本的样式插入链接与图片如何插入一段漂亮的代码片生成一个适合你的列表创建一个表格设定内容居中、居左、居右SmartyPants创建一个自定义列表如何创建一个注…

【C语言从0到1之文件操作(FILE)】(原理 画图 举例 不信教不会你 不要放收藏夹落灰 学起来好嘛)

🕺作者:迷茫的启明星 🎃专栏:《数据库》《C语言从0到1专栏》《数据结构》《C语言杂谈》🏇分享喜欢的一句话:心如花木,向阳而生前言在我们的学习中,文件操作是被我们忽略,…

NodeJS 与第三方模块 mysql(基本操作)

文章目录参考描述mysql 模块连接数据库检测基本操作查询数据与代码分离原则占位符插入另一种姿态修改另一种姿态删除标记删除参考 项目描述哔哩哔哩黑马程序员搜索引擎Bing 描述 项目描述NodeJSv18.13.0nodemon2.0.20MySQL5.7.40mysql2.18.1 mysql 模块 npm(Node…

Linux——进程

目录 冯诺依曼体系结构 操作系统(Operator System) 概念 设计OS的目的 定位 如何理解 "管理" 总结 系统调用和库函数概念 承上启下 进程 基本概念 描述进程-PCB task_struct-PCB的一种 task_ struct内容分类 组织进程 查看进程 通过系统调用获取进程…

Pycharm使用Git进行版本控制(自建远端Git仓库)

目录本地Git安装远端Git仓库搭建在Pycharm中使用Git进行版本控制设置Git可执行文件路径创建本地Git仓库设置远端Git仓库提交及推送本地Git安装 安装本地Git用于被Pycharm调用,安装方法参考以下博客: Git 的下载与安装_作者:fengzhx0820 远端…

四轮两驱小车(四):STM32驱动5路灰度传感器PID循迹

目录 前言: 小车效果展示: 5路数字灰度传感器: 巡线思路: 加入PID调节的代码: 前言: 之前买了一批5路灰度传感器,想用这传感器进行循迹,无奈网上和官方的资料提供的还是比较少&a…

ARM X210 官方 uboot 配置编译实践

一、X210官方uboot配置编译实践1 1. 找到官方移植好的 uboot(BSP 概念) (1) 源头的源代码是 uboot 官网下载的。这个下载的源代码可能没有你当前使用的开发板的移植,甚至找不到当前开发板使用的 SoC 对应的移植版本。 (2) SoC 厂商在推出一…

分享145个ASP源码,总有一款适合您

ASP源码 分享145个ASP源码,总有一款适合您 下面是文件的名字,我放了一些图片,文章里不是所有的图主要是放不下..., 145个ASP源码下载链接:https://pan.baidu.com/s/1gxm3rFFLu8pUhVncQga6-g?pwd7n85 提取码&#x…

HJ56、HJ58、JZ4、JZ6、JZ15、JZ17几道题

文章目录HJ56 完全数计算题目描述:具体实现:HJ58 输入n个整数,输出其中最小的k个题目描述:具体实现:JZ4 二维数组中的查找题目描述:具体实现:JZ6 从尾到头打印链表题目描述:具体实现…