迭代反投影法代码_Ceres求解直接法BA实现自动求导

作者:郭田峰

来源:公众号@3D视觉工坊

BA,即Bundle Adjustment,通常译为光束法平差,束调整,捆绑调整等。但高翔博士觉得这些译名不如英文名称来得直观,所以保留英文名,简称BA。

所谓BA,是指从视觉图像中提炼出最优的3D模型和相机参数。在视觉SLAM里,BA特征点法和直接法两种。前者是最小化重投影误差作为优化目标,后者是以最小化光度误差为目标。

对于特征点法BA,高翔博士所著的《视觉SLAM十四讲》第二版第九章作了非常详细的说明。对于直接法BA,在深蓝学院的课程《视觉SLAM理论与实践》中有用g2o求解的习题,但没有提到Ceres求解。而且,习题中给出的是双线性插值来得到图像点的灰度值。我们知道,直接法BA需要判断图像边界,而且Ceres对双线性插值是不能自动求导的。这都会增加代码实现的难度。

在课后作业里有一题要求用g2o实现直接法BA。相对来说g2o来说,我个人更喜欢用Ceres,毕竟Ceres是谷歌出品,而且,谷歌的非线性优化大多是用Ceres来解决的,功能和效率应该是值得我们信任的。

我们知道,Ceres是推荐我们尽可能使用自动求导的,一是准确性更有保障;二是求解更快速。所以,我们要寻找能实现自动求导的实现方法。

Ceres提供的Ceres的Grid2D和BiCubicInterpolator联合使用可以解决上述两个问题。Grid2D和BiCubicInterpolator的使用需要包含头文件cubic_interpolation.h。

Grid2D是无限二维网格的对象,可以用来载入图像数据,如果是灰度图,其值用标量,如果是彩色图像,其值用3维向量。由于网格的输入数据总是有限的,而网格本身是无限的,因为需要通过使用双三次插值BiCubicInterpolator来计算网格之间的值。而超出网格范围,则将返回最近边缘的值。

将灰度图像数据加载到Grid2D对象,可以避免我们在代码中判断图像边界的问题。而且,Grid2D还可以利用BiCubicInterpolator实现双三次插值,它相对于双线性插值的优点是能实现自动求导。利用它们写出的代码更简洁,执行效率也更高。

Grid2D对象和BiCubicInterpolator对象的定义:

std::unique_ptr<:grid2d> > 变量名;std::unique_ptr<:bicubicinterpolator> > > 变量

数据类型一般是简单类型,如int,float,double等,上面两个定义的数据类型和数据维数必须相同。数据维数是指值是几维数据,默认值为1,即函数值为标量时可以不指定该参数。定义好了对象变量,就可以初始化了,格式如下:

Grid2D对象.reset(new ceres::Grid2D(数据首地址, 首行号, 总行数, 首列号, 总列数));BiCubicInterpolator对象.reset(new ceres::BiCubicInterpolator<:grid2d> >(* Grid2D对象))

数据一般使用vector类型以及嵌套,对于灰度图,我使用的是vector>。

Grid2D还有两个参数,分别是表示数据的储存顺序为行还是列,以及值为向量时向量的每一维值的存储顺序是行还是列,由于在本文中并不重要,所以这里不作介绍。代码中直接采用了默认值。

计算残差代码如下:

struct GetPixelGrayValue {    GetPixelGrayValue(const float pixel_gray_val_in[16],                      const int rows,                      const int cols,                      const std::vector& vec_pixel_gray_values)    {        for (int i = 0; i < 16; i++)            pixel_gray_val_in_[i] = pixel_gray_val_in[i];        rows_ = rows;        cols_ = cols;                // 初始化grid2d        grid2d.reset(new ceres::Grid2D(            &vec_pixel_gray_values[0], 0, rows_, 0, cols_));                //双三次插值        get_pixel_gray_val.reset(            new ceres::BiCubicInterpolator<:grid2d> >(*grid2d));    }        template bool operator()(const T* const so3t,              //模型参数,位姿,6维const T* const xyz,             //模型参数,3D点,3维T* residual ) const              //残差,4*4图像块,16维{        // 计算变换后的u和v        T u_out, v_out, pt[3], r[3];        r[0] = so3t[0];        r[1] = so3t[1];        r[2] = so3t[2];        ceres::AngleAxisRotatePoint(r, xyz, pt);        pt[0] += so3t[3];        pt[1] += so3t[4];        pt[2] += so3t[5];        u_out = pt[0] * T(fx) / pt[2] + T(cx);        v_out = pt[1] * T(fy) / pt[2] + T(cy);                        for (int i = 0; i < 16; i++)        {            int m = i / 4;            int n = i % 4;            T u, v, pixel_gray_val_out;            u = u_out + T(m - 2);            v = v_out + T(n - 2);            get_pixel_gray_val->Evaluate(u, v, &pixel_gray_val_out);            residual[i] = T(pixel_gray_val_in_[i]) - pixel_gray_val_out;        }                return true;    }    float pixel_gray_val_in_[16];    int rows_,cols_;    std::unique_ptr<:grid2d> > grid2d;     std::unique_ptr<:bicubicinterpolator> > > get_pixel_gray_val;};

添加残差块的代码:

for (int ip = 0; ip < poses_num; ip++){ for (int jp = 0; jp < points_num; jp++) { double *pose_position = first_poses_pos + ip * 6; double *point_position = first_point_pos + jp * 3; float gray_values[16]{}; for ( int i = 0; i < 16; i++ ) gray_values[i] = patch_gray_values[jp][i]; problem.AddResidualBlock( new ceres::AutoDiffCostFunction ( new GetPixelGrayValue ( gray_values, multi_img_rows_cols[ip * 2], multi_img_rows_cols[ip * 2 + 1], multi_img_gray_values[ip] ) ), new ceres::HuberLoss(1.0), pose_position, point_position ); }}

这里有必要就说明的是,要判定变换后的u和v是否在图像内,如果超界了,则该组数据弃之不用。在使用Grid2D和BiCubicInterpolator后,超界后使用的值是最接近的边缘的值。这两者处理结果看似差别很大,但对结果影响很小的,几乎可以忽略不计。

下面的原来的题目:

57803ec50c7cf7d9221bdc64c0548112.png
8795045215e366eaa554f2427e1dd386.png

对于相同的数据,g2o和Ceres求解执行结果如下。从截图数据显示,Ceres优化一共进行了33次迭代,用时7秒多;g2o优化一共进行了199次迭代,用时64秒左右。在这个优化题目里,两者差异非常明显。

c8505917de0ece0249b137015fa8a4d0.png

g2o求解直接法BA执行结果截图

85c66bf66d52ec910558117f35bc1ef8.png

Ceres求解直接法BA执行结果截图

在公众号后台回复「DirectBA」,获取g2o和Ceres的求解代码。

本文仅做学术分享,如有侵权,请联系删文。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值