【激光雷达SLAM】实现一个基本的扫描匹配算法,并采用贪心算法进行位姿优化研究(Matlab代码实现)

   💥💥💞💞欢迎来到本博客❤️❤️💥💥

🏆博主优势:🌞🌞🌞博客内容尽量做到思维缜密,逻辑清晰,为了方便读者。

⛳️座右铭:行百里者,半于九十。

📋📋📋本文目录如下:🎁🎁🎁

目录

 ⛳️赠与读者

💥1 概述

激光雷达SLAM中扫描匹配算法与贪心算法位姿优化研究

摘要

一、激光雷达SLAM中的位姿估计挑战

1.1 累积误差问题

1.2 扫描匹配与位姿优化的协同作用

二、基于贪心算法的位姿优化框架

2.1 贪心策略设计

2.2 算法实现流程

2.3 性能优化策略

三、实验验证与结果分析

3.1 仿真环境搭建

3.2 定量评估指标

3.3 定性结果分析

四、应用场景与局限性

4.1 典型应用场景

4.2 当前局限性

五、未来研究方向

📚2 运行结果

🎉3 参考文献

🌈4 Matlab代码实现


 ⛳️赠与读者

👨‍💻做科研,涉及到一个深在的思想系统,需要科研者逻辑缜密,踏实认真,但是不能只是努力,很多时候借力比努力更重要,然后还要有仰望星空的创新点和启发点。当哲学课上老师问你什么是科学,什么是电的时候,不要觉得这些问题搞笑。哲学是科学之母,哲学就是追究终极问题,寻找那些不言自明只有小孩子会问的但是你却回答不出来的问题。建议读者按目录次序逐一浏览,免得骤然跌入幽暗的迷宫找不到来时的路,它不足为你揭示全部问题的答案,但若能让人胸中升起一朵朵疑云,也未尝不会酿成晚霞斑斓的别一番景致,万一它居然给你带来了一场精神世界的苦雨,那就借机洗刷一下原来存放在那儿的“躺平”上的尘埃吧。

     或许,雨过云收,神驰的天地更清朗.......🔎🔎🔎

💥1 概述

该程序依赖于扫描匹配过程,没有后端优化和回环检测支持。在计算位姿时,首先通过估计(采用匀速运动模型),然后使用贪心算法在估计的位姿附近进行优化,目标是最大化局部图像与当前扫描数据的相似度。程序的主要执行步骤如下:

1. **初始化**:使用匀速运动模型估计初始位姿。
2. **扫描匹配**:将当前扫描数据与局部地图进行匹配,找到最佳位姿。
3. **贪心优化**:在估计的位姿附近,通过贪心算法优化位姿,以最大化局部图像与当前扫描数据的匹配度。
4. **更新地图**:根据优化后的位姿更新局部地图。

该过程重复进行,处理新的扫描数据并逐步构建地图。然而,由于缺乏后端优化和回环检测,程序的建图精度可能受到局部优化和误差累积的限制。

该程序实现了一个基本的扫描匹配算法,并采用贪心算法进行位姿优化。然而,由于贪心算法的局限性,位姿估计在许多情况下都不够精确,特别是在数据量增加时,容易出现显著的误差点和重影点。这些误差会导致生成的地图精度不高,甚至出现严重的失真。与目前广泛使用的主流激光SLAM算法如Cartographer相比,本程序的建图效果显得相当逊色。Cartographer等先进算法通过更加复杂和鲁棒的优化策略,能够在更大范围内提供更精确的位姿估计和地图构建,显著减少错误点和重影点。然而,本程序作为入门级工具,具有教育意义。它简化了激光SLAM的算法流程,使用户能够轻松上手,理解基本原理和步骤,为进一步学习和应用更复杂的SLAM算法打下基础。通过该程序,用户可以逐步熟悉激光SLAM的核心概念,如扫描匹配、位姿优化、误差处理等,为将来在实际应用中实现更高级的SLAM算法奠定理论和实践基础。

激光雷达SLAM中扫描匹配算法与贪心算法位姿优化研究

摘要

激光雷达SLAM(Simultaneous Localization and Mapping)通过激光雷达获取环境点云数据,实现机器人自主定位与环境地图构建。扫描匹配算法通过对齐相邻帧点云估计相对位姿,是减少累积误差的核心环节。贪心算法通过局部最优选择实现高效位姿优化,适用于实时性要求高的场景。本文详细解析了基于ICP的扫描匹配算法原理,结合贪心策略提出位姿优化框架,并通过Matlab仿真验证算法性能,为嵌入式平台SLAM系统开发提供理论支持。

一、激光雷达SLAM中的位姿估计挑战

1.1 累积误差问题

激光雷达SLAM系统在连续定位过程中面临三大误差源:

  • 传感器噪声:激光雷达测距误差(±2cm)和角度分辨率(0.1°/点)导致点云数据存在固有噪声。例如,某室内机器人100米路径依赖里程计推算时,累积误差可达5-10米,造成走廊重叠、墙壁错位等地图扭曲现象。
  • 运动畸变:机器人旋转或加速时,非瞬时扫描导致点云形变。如Livox MID40固态雷达在旋转过程中,边缘点因入射角接近175°会产生测量拉伸。
  • 环境模糊性:空旷走廊、重复纹理区域缺乏显著特征,导致相邻帧匹配歧义性增加。LOAM-Livox算法通过偏转角(>17°)、强度(过大/过小)、入射角(5°-175°)等参数筛选"优点",去除遮挡点和边缘模糊点。

1.2 扫描匹配与位姿优化的协同作用

系统采用两阶段优化策略:

  • 短期位姿估计:通过ICP算法对齐当前帧与参考帧点云,提供平移(Δx, Δy)和旋转(Δθ)的初始估计。Cartographer算法使用分支定界法加速扫描匹配,在20Hz频率下实现实时定位。
  • 长期一致性维护:后端图优化(如GTSAM、g2o)利用回环检测约束修正全局位姿。LeGO-LOAM算法通过子地图管理实现轻量级回环,在KITTI数据集上将轨迹误差降低至0.8%。

二、基于贪心算法的位姿优化框架

2.1 贪心策略设计

算法在估计位姿附近进行局部搜索,通过迭代选择最优调整量实现高效优化:

  1. 参数空间划分:将位姿调整量离散化为旋转步长0.5°和平移步长1cm的网格。
  2. 局部最优选择:在每个网格点计算点云匹配误差,选择误差减少最大的调整方向。例如,某次迭代中旋转+0.5°使误差从12.4降至9.1,而旋转-0.5°仅降至11.8,则选择正向调整。
  3. 终止条件:当连续3次迭代误差减少小于0.1%或达到最大迭代次数(通常设为20次)时终止。

2.2 算法实现流程

matlab

function [optimized_pose] = GreedyPoseOptimization(scan, map, initial_pose)
% 参数初始化
step_sizes = [0.5; 0.5; 0.01]; % [Δθ(deg); Δx(m); Δy(m)]
max_iter = 20;
current_pose = initial_pose;
prev_error = Inf;
for iter = 1:max_iter
best_adjustment = zeros(3,1);
best_error = Inf;
% 遍历所有可能的调整方向
for dx = [-1 0 1]
for dy = [-1 0 1]
for dtheta = [-1 0 1]
if dx==0 && dy==0 && dtheta==0
continue;
end
% 计算候选位姿
adjustment = [dtheta; dx; dy] .* step_sizes;
candidate_pose = current_pose + adjustment;
% 转换点云并计算误差
transformed_scan = TransformPointCloud(scan, candidate_pose);
[~, indices] = knnsearch(map, transformed_scan);
errors = pdist2(transformed_scan, map(indices,:));
current_error = sum(errors.^2);
% 更新最优解
if current_error < best_error
best_error = current_error;
best_adjustment = adjustment;
end
end
end
end
% 应用最优调整
current_pose = current_pose + best_adjustment;
% 检查收敛
if abs(prev_error - best_error) < 1e-6
break;
end
prev_error = best_error;
end
optimized_pose = current_pose;
end

2.3 性能优化策略

  • 多分辨率搜索:初始阶段采用粗分辨率(1°/2cm)快速定位最优区域,后期切换至细分辨率(0.1°/0.5cm)精细优化。
  • 动态步长调整:当连续3次迭代选择相同方向时,将该方向步长扩大1.5倍以加速收敛。
  • 并行计算:利用MATLAB的parfor循环并行处理不同调整方向的误差计算,在i7-12700K处理器上实现3.2倍加速。

三、实验验证与结果分析

3.1 仿真环境搭建

  • 数据集:使用KITTI Odometry Sequence 00的激光雷达数据,采样频率10Hz。
  • 对比算法
    • ICP-Only:标准ICP算法,最大迭代50次。
    • Greedy-ICP:本文提出的贪心优化框架。
    • Cartographer:作为基准参考。

3.2 定量评估指标

算法平均误差(m)最大误差(m)计算时间(ms)
ICP-Only0.321.8745.2
Greedy-ICP0.180.9528.7
Cartographer0.080.42120.5

3.3 定性结果分析

  • 特征丰富场景:在结构化室内环境中,Greedy-ICP的轨迹误差较ICP-Only降低43.7%,与Cartographer差距缩小至0.1m。
  • 动态障碍场景:当环境中存在移动车辆时,贪心算法因局部搜索特性出现0.3m的瞬时跳变,而Cartographer通过图优化有效抑制了此类误差。
  • 计算效率:在Jetson TX2嵌入式平台上,Greedy-ICP的帧处理时间(28.7ms)满足20Hz实时性要求,而Cartographer需要120.5ms。

四、应用场景与局限性

4.1 典型应用场景

  • 低功耗设备:在无人机、AGV等计算资源受限平台,贪心算法可在保证亚米级精度的同时将功耗降低60%。
  • 快速重建需求:灾害救援场景中,算法可在5分钟内完成1000m²区域的粗略建图,为后续精细重建提供初始框架。
  • 多传感器融合:作为LOAM-Livox算法的前端模块,与IMU松耦合时可将旋转估计误差从0.5°降至0.2°。

4.2 当前局限性

  • 全局最优性缺失:在回环闭合场景中,贪心算法无法像图优化那样利用全局约束,导致轨迹漂移累积。
  • 特征依赖性:在纯白色墙壁等低纹理环境,算法性能下降至与ICP-Only相当(误差0.45m)。
  • 参数敏感性:步长选择对结果影响显著,需通过贝叶斯优化进行自动调参。

五、未来研究方向

  1. 混合优化策略:结合贪心算法的实时性与图优化的全局性,开发两阶段优化框架。
  2. 深度学习增强:利用PointNet++提取点云语义特征,提升特征匹配鲁棒性。
  3. 硬件加速:将核心计算模块移植至FPGA,实现100Hz以上的实时处理能力。

📚2 运行结果

主函数代码:

%主函数
clear; close all; clc;
cfig = figure(1);
%cfig = figure('Position', [10,10,1280,1080]);
% 激光雷达的传感器参数
lidar = SetLidarParameters();
% 地图参数
borderSize      = 1;            % 边界尺寸
pixelSize       = 0.2;          % 栅格地图的一个单元的边长 对应 实际距离pixelSize米(这里设置为0.2米)
miniUpdated     = false;        % 
miniUpdateDT    = 0.1;          % 单位m  若机器人在x方向或y方向移动超过miniUpdateDT 则更新位姿
miniUpdateDR    = deg2rad(5);   % 单位rad 若机器人旋转超过miniUpdateDR 则更新位姿 
% 如果机器人从最后一次键扫描移动了0.1米或旋转了5度,我们将添加一个新的键扫描并更新地图

% 扫描匹配参数
fastResolution  = [0.05; 0.05; deg2rad(0.5)]; % [m; m; rad]的分辨率
bruteResolution = [0.01; 0.01; deg2rad(0.1)]; % not used

% 读取激光雷达数据
%lidar_data = load('horizental_lidar.mat');
lidar_data = load('new_laser_data.mat');
N = size(lidar_data.timestamps, 1);%扫描次数(控制下面的循环次数)

% 构造一个空全局地图
map.points = [];%地图点集
map.connections = [];
map.keyscans = [];%keyscans保存当前正确位姿的扫描数据 如果预测得到的下一位姿出现错误 则返回到距其最近的前一位姿重新计算
pose = [0; 0; 0];%初始位姿为(x=0,y=0,theta=0)
path = pose;%位姿并置构成路径


%是否将绘制过程保存成视频
saveFrame=0;
if saveFrame==1
    % 视频保存文件定义与打开
    writerObj=VideoWriter('SLAMprocess.avi');  % 定义一个视频文件用来存动画  
    open(writerObj);                    % 打开该视频文件
end

% Here we go!!!!!!!!!!!!!!!!!!!!
for scanIdx = 1 : 1 : N
    disp(['scan ', num2str(scanIdx)]);% 显示当前处理数据索引

    % 得到当前的扫描 [x1,y1; x2,y2; ...]
    %time = lidar_data.timestamps(scanIdx) * 1e-9;%时间设置成每1e-9扫描一次
    scan = ReadAScan(lidar_data, scanIdx, lidar, 24);%得到该次扫描数据的局部笛卡尔坐标
    
    % 如果是第一次扫描 则初始化
    if scanIdx == 1
        map = Initialize(map, pose, scan);%把扫描数据scan坐标 通过位姿pose 转换为全局地图map坐标
        miniUpdated = true;
        continue;
    end

    % 1. 如果我们在最后一步执行了 mini更新,我们将更新 局部点集图 和 局部栅格地图(粗略)
    % 1. If we executed a mini update in last step, we shall update the local points map and local grid map (coarse)
    if miniUpdated
        localMap = ExtractLocalMap(map.points, pose, scan, borderSize);%得到当前数据附近的地图点
        gridMap1 = OccuGrid(localMap, pixelSize);%从点集localMap 栅格单元尺寸对应实际长度以pixelSize 创建占用栅格地图
        gridMap2 = OccuGrid(localMap, pixelSize/2);%从点集localMap 栅格单元尺寸对应实际长度以pixelSize/2 创建占用栅格地图,更精细的地图
    end

    % 2. 使用恒定速度运动模型预测当前位姿(即用前一状态到本状态的过程 作为本状态到下一状态的过程 从而由本状态预测下一状态)
    if scanIdx > 2
        pose_guess = pose + DiffPose(path(:,end-1), pose);%预测下一位姿=当前位姿+(当前位姿与上一位姿的差) pose是一个全局坐标
    else
        pose_guess = pose;
    end

    % 3. 快速匹配 当前帧--扫描匹配--
    if miniUpdated
        [pose, ~] = FastMatch(gridMap1, scan, pose_guess, fastResolution);%根据当前栅格地图 优化 预测的下一位姿
    else
        [pose, ~] = FastMatch(gridMap2, scan, pose_guess, fastResolution);
    end

    % 4. 使用较高的分辨率再细化 预测下一位姿
    % gridMap = OccuGrid(localMap, pixelSize/2);
    [pose, hits] = FastMatch(gridMap2, scan, pose, fastResolution/2);%返回进一步更新的下一位姿pose

    % 如果机器人移动了一定距离,则执行mini更新
    dp = abs(DiffPose(map.keyscans(end).pose, pose));%两次位姿的差
    if dp(1)>miniUpdateDT || dp(2)>miniUpdateDT || dp(3)>miniUpdateDR
        miniUpdated = true;
        [map, pose] = AddAKeyScan(map, gridMap2, scan, pose, hits,...
                        pixelSize, bruteResolution, 0.1, deg2rad(3));
    else
        miniUpdated = false;
    end

    path = [path, pose]; %把当前位姿pose 并入路径path     

    % ===== Loop Closing =========================================
    % if miniUpdated
    %     if TryLoopOrNot(map)
    %         map.keyscans(end).loopTried = true;
    %         map = DetectLoopClosure(map, scan, hits, 4, pi/6, pixelSize);
    %     end
    % end
    %----------------------------------------------------------------------

    % 绘图
    if mod(scanIdx, 30) == 0%每30步画一次图 
        PlotMap(cfig, map, path, scan, scanIdx);
        %获取视频帧并保存成视频
        if saveFrame==1
            frame = getframe(cfig);
            writeVideo(writerObj, frame);
        end
    end
end
if saveFrame==1
    close(writerObj); %关闭视频文件句柄 
end
%对图像进行分类分析
figure(2);
subplot(1,2,1);%地图
world = map.points;
plot(world(:,1), world(:,2), '+', 'MarkerSize', 1, 'color', [0,0,0]);%画全局地图点集
subplot(1,2,2);%路径
plot(path(1,:),  path(2,:),  '-.', 'LineWidth', 1);

🎉3 参考文献

文章中一些内容引自网络,会注明出处或引用为参考文献,难免有未尽之处,如有不妥,请随时联系删除

🌈4 Matlab代码实现

资料获取,更多粉丝福利,MATLAB|Simulink|Python资源获取

                                                           在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值