IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 人工智能 -> vins estimator Td factor -> 正文阅读

[人工智能]vins estimator Td factor

简介

相加快门

  • 快门是照相机用来控制感光片有效曝光时间的机构,Global Shutter(全局快门)与Rolling Shutter(卷帘快门)对应全局曝光和卷帘曝光模式,这两种曝光模式都是相机常见的曝光模式。
  • Global Shutter 特点
    • 感光元件的所有像素点同时曝光一定时间,进而成像。
    • 曝光时间比Rolling Shutter短,曝光整帧后输出像素数据。
    • 对于高速运动的部分曝光后体现出模糊的现象。
    • 一般使用CCD作为Sensor,对像素点输出的带宽的要求较高,CCD的造价也相对CMOS较高。
  • Rolling Shutter 特点
    • 感光元件的所有像素点逐行轮流曝光一定时间,进而成像。
    • 会出现“果冻现象”。
    • 曝光时间比Global Shutter长,但是曝光一行输出一行。
    • 对于高速运动的部分曝光后体现出弯曲的现象。
    • 一般使用CMOS作为Sensor,对像素点输出的带宽的要求较低,CMOS的造价也相对较低。

本部分功能

  • 对imu-camera 时间戳不完全同步和 Rolling shutter 相机的支持
  • 该部分的功能是优化 IMU 与相机硬件的固定时间差
  • 认为滑窗内 imu 与相机 硬件的时间间隔一样。
  • 输入的变量:
    • i 时刻 角点 (j时刻角点一样,赋值一份)
      • 在归一化平面的坐标 pts_i
      • 角点在归一化平面的速度 velocity_i
      • 处理IMU数据时用到的时间同步误差 td_i
      • 角点在归一化平面 行坐标 _row_i = _row_i - ROW / 2
  • 每张图片以中心时间曝光作为其时间。

残差

优化变量

  • imu_i 的全局位姿 维度:7
  • imu_j 的全局位姿 维度:7
  • 外参 维度:7
  • inv_dep_i 维度:1
  • 时间差 变量 para_Td[0] 维度:1

残差计算

  • 角点i在归一化平面的row坐标
    • pts_i - (td - td_i + TR / ROW * row_i) * velocity_i
    • pts_camera_i = pts_i_td / inv_dep_i
  • 通过外参转换到 imu坐标系
    • pts_imu_i = qic * pts_camera_i + tic;
  • 通过imu_i的全局位姿,将角点转换到世界坐标系
    • pts_w = Qi * pts_imu_i + Pi;
  • 将角点通过 imu_j 转换到 j时刻imu坐标系下位姿
    • pts_imu_j = Qj.inverse() * (pts_w - Pj);
  • 转换到 j 时刻 相机坐标系下
    • pts_camera_j = qic.inverse() * (pts_imu_j - tic);

求解其Jacobian

  • 求解 jacobian时,即残差对 优化变量 求导
  • 求导分两步,先对 pts_camera_j 求导,再对 变量求导。

对变量分别求偏导

对 pts_camera_j 求导

  • 残差对 pts_camera_j求导

  • 非单位球面误差时:
    r e d u c e = [ 1. / d e p j 0 ? p t s _ c a m e r a _ j ( 0 ) / ( d e p _ j ? d e p _ j ) 0 1. / d e p j ? p t s _ c a m e r a _ j ( 0 ) / ( d e p _ j ? d e p _ j ) ] { {reduce} = \begin{bmatrix} 1./dep_j & 0 & -{pts\_camera\_j(0) / (dep\_j * dep\_j)} \\ 0 & 1./dep_j & -{pts\_camera\_j(0) / (dep\_j * dep\_j)} \end{bmatrix}} reduce=[1./depj?0?01./depj???pts_camera_j(0)/(dep_j?dep_j)?pts_camera_j(0)/(dep_j?dep_j)?]

  • 由于权重(信息矩阵)因素,故 reduce = sqrt_info * reduce

        if (jacobians)
        {
            Eigen::Matrix3d Ri = Qi.toRotationMatrix();
            Eigen::Matrix3d Rj = Qj.toRotationMatrix();
            Eigen::Matrix3d ric = qic.toRotationMatrix();
            Eigen::Matrix<double, 2, 3> reduce(2, 3);
    #ifdef UNIT_SPHERE_ERROR
            double norm = pts_camera_j.norm();
            Eigen::Matrix3d norm_jaco;
            double x1, x2, x3;
            x1 = pts_camera_j(0);
            x2 = pts_camera_j(1);
            x3 = pts_camera_j(2);
            norm_jaco << 1.0 / norm - x1 * x1 / pow(norm, 3), - x1 * x2 / pow(norm, 3),            - x1 * x3 / pow(norm, 3),
                         - x1 * x2 / pow(norm, 3),            1.0 / norm - x2 * x2 / pow(norm, 3), - x2 * x3 / pow(norm, 3),
                         - x1 * x3 / pow(norm, 3),            - x2 * x3 / pow(norm, 3),            1.0 / norm - x3 * x3 / pow(norm, 3);
            reduce = tangent_base * norm_jaco;
    #else
            reduce << 1. / dep_j, 0, -pts_camera_j(0) / (dep_j * dep_j),
                0, 1. / dep_j, -pts_camera_j(1) / (dep_j * dep_j);
    #endif
    

对 Pose_i 求导

  • pts_camera_jPose_i求导,对pts_camera_j偏导乘以其值
  • pts_camera_jP_i的偏导: q i c ? 1 Q j ? 1 {q_{ic}^{-1} Q_j^{-1}} qic?1?Qj?1?
  • pts_camera_jQ_i的偏导: q i c ? 1 Q j ? 1 ? Q i ( p t s _ i m u _ i ) ∧ {q_{ic}^{-1} Q_j^{-1} -Q_i ({pts\_imu\_i})^\wedge} qic?1?Qj?1??Qi?(pts_imu_i) (通过右扰动求导)

对 Pose_j 求导

  • pts_camera_jPose_j求导,对pts_camera_j偏导乘以其值
  • pts_camera_jP_j的偏导: q i c ? 1 Q j ? 1 {q_{ic}^{-1} Q_j^{-1}} qic?1?Qj?1?
  • pts_camera_jQ_j的偏导: q i c ? 1 ( p t s _ i m u _ j ) ∧ {q_{ic}^{-1} ({pts\_imu\_j})^\wedge} qic?1?(pts_imu_j) (通过右扰动求导)

对 外参 求导

  • pts_camera_jPose_{ic}求导,对pts_camera_j偏导乘以其值
  • pts_camera_jP_{ic}的偏导: q i c ? 1 ( Q j ? 1 Q i ? I ) {q_{ic}^{-1} (Q_j^{-1}Q_i -I)} qic?1?(Qj?1?Qi??I)
  • pts_camera_jQ_{ic}的偏导: q i c ? 1 ( p t s _ i m u _ j ? t i c ) ∧ + q i c ? 1 Q j ? 1 Q i q i c ( p t s _ c a m e r a _ i ? t i c ) ∧ {q_{ic}^{-1}({pts\_imu\_j}-t_{ic})^\wedge + q_{ic}^{-1}Q_j^{-1} Q_iq_{ic}({pts\_camera\_i}-t_{ic})^\wedge} qic?1?(pts_imu_j?tic?)+qic?1?Qj?1?Qi?qic?(pts_camera_i?tic?) (通过右扰动求导)

对特征点深度 inv_dep_i 求导

  • pts_camera_jinv_dep_i求导,对pts_camera_j偏导乘以其值
  • ? q i c ? 1 Q j ? 1 Q i q i c p t s _ i _ t d / ( i n v _ d e p _ i ) 2 {-q_{ic}^{-1}Q_j^{-1} Q_iq_{ic} {pts\_i\_td} / (inv\_dep\_i)^2} ?qic?1?Qj?1?Qi?qic?pts_i_td/(inv_dep_i)2

对时间差 求导

  • 对时间差求偏导时, pts_camera_jpts_camera_j都有其变量,故:
  • pts_camera_j 对 dt 求导: r e d u c e ? ? q i c ? 1 Q j ? 1 Q i q i c p t s _ i _ t d ? v e l o c i t y _ y / ( i n v _ d e p _ i ) { reduce * -q_{ic}^{-1}Q_j^{-1} Q_iq_{ic} {pts\_i\_td} * velocity\_y/ (inv\_dep\_i) } reduce??qic?1?Qj?1?Qi?qic?pts_i_td?velocity_y/(inv_dep_i)
  • pts_camera_j 对 dt 求导: s q r t _ i n f o ? v e l o c i t y _ y {sqrt\_info *velocity\_y } sqrt_info?velocity_y

代码实现

bool ProjectionTdFactor::Evaluate(double const *const *parameters, double *residuals, double **jacobians) const
{
    TicToc tic_toc;
    Eigen::Vector3d Pi(parameters[0][0], parameters[0][1], parameters[0][2]);
    Eigen::Quaterniond Qi(parameters[0][6], parameters[0][3], parameters[0][4], parameters[0][5]);

    Eigen::Vector3d Pj(parameters[1][0], parameters[1][1], parameters[1][2]);
    Eigen::Quaterniond Qj(parameters[1][6], parameters[1][3], parameters[1][4], parameters[1][5]);

    Eigen::Vector3d tic(parameters[2][0], parameters[2][1], parameters[2][2]);
    Eigen::Quaterniond qic(parameters[2][6], parameters[2][3], parameters[2][4], parameters[2][5]);

    double inv_dep_i = parameters[3][0];

    double td = parameters[4][0];

    Eigen::Vector3d pts_i_td, pts_j_td;

    pts_i_td = pts_i - (td - td_i + TR / ROW * row_i) * velocity_i;
    pts_j_td = pts_j - (td - td_j + TR / ROW * row_j) * velocity_j;
    Eigen::Vector3d pts_camera_i = pts_i_td / inv_dep_i;
    Eigen::Vector3d pts_imu_i = qic * pts_camera_i + tic;
    Eigen::Vector3d pts_w = Qi * pts_imu_i + Pi;
    Eigen::Vector3d pts_imu_j = Qj.inverse() * (pts_w - Pj);
    Eigen::Vector3d pts_camera_j = qic.inverse() * (pts_imu_j - tic);
    Eigen::Map<Eigen::Vector2d> residual(residuals);

#ifdef UNIT_SPHERE_ERROR 
    residual =  tangent_base * (pts_camera_j.normalized() - pts_j_td.normalized());
#else
    double dep_j = pts_camera_j.z();
    residual = (pts_camera_j / dep_j).head<2>() - pts_j_td.head<2>();
#endif

    residual = sqrt_info * residual;

    if (jacobians)
    {
        Eigen::Matrix3d Ri = Qi.toRotationMatrix();
        Eigen::Matrix3d Rj = Qj.toRotationMatrix();
        Eigen::Matrix3d ric = qic.toRotationMatrix();
        Eigen::Matrix<double, 2, 3> reduce(2, 3);
#ifdef UNIT_SPHERE_ERROR
        double norm = pts_camera_j.norm();
        Eigen::Matrix3d norm_jaco;
        double x1, x2, x3;
        x1 = pts_camera_j(0);
        x2 = pts_camera_j(1);
        x3 = pts_camera_j(2);
        norm_jaco << 1.0 / norm - x1 * x1 / pow(norm, 3), - x1 * x2 / pow(norm, 3),            - x1 * x3 / pow(norm, 3),
                     - x1 * x2 / pow(norm, 3),            1.0 / norm - x2 * x2 / pow(norm, 3), - x2 * x3 / pow(norm, 3),
                     - x1 * x3 / pow(norm, 3),            - x2 * x3 / pow(norm, 3),            1.0 / norm - x3 * x3 / pow(norm, 3);
        reduce = tangent_base * norm_jaco;
#else
        reduce << 1. / dep_j, 0, -pts_camera_j(0) / (dep_j * dep_j),
            0, 1. / dep_j, -pts_camera_j(1) / (dep_j * dep_j);
#endif
        reduce = sqrt_info * reduce;

        if (jacobians[0])
        {
            Eigen::Map<Eigen::Matrix<double, 2, 7, Eigen::RowMajor>> jacobian_pose_i(jacobians[0]);

            Eigen::Matrix<double, 3, 6> jaco_i;
            jaco_i.leftCols<3>() = ric.transpose() * Rj.transpose();
            jaco_i.rightCols<3>() = ric.transpose() * Rj.transpose() * Ri * -Utility::skewSymmetric(pts_imu_i);

            jacobian_pose_i.leftCols<6>() = reduce * jaco_i;
            jacobian_pose_i.rightCols<1>().setZero();
        }

        if (jacobians[1])
        {
            Eigen::Map<Eigen::Matrix<double, 2, 7, Eigen::RowMajor>> jacobian_pose_j(jacobians[1]);

            Eigen::Matrix<double, 3, 6> jaco_j;
            jaco_j.leftCols<3>() = ric.transpose() * -Rj.transpose();
            jaco_j.rightCols<3>() = ric.transpose() * Utility::skewSymmetric(pts_imu_j);

            jacobian_pose_j.leftCols<6>() = reduce * jaco_j;
            jacobian_pose_j.rightCols<1>().setZero();
        }
        if (jacobians[2])
        {
            Eigen::Map<Eigen::Matrix<double, 2, 7, Eigen::RowMajor>> jacobian_ex_pose(jacobians[2]);
            Eigen::Matrix<double, 3, 6> jaco_ex;
            jaco_ex.leftCols<3>() = ric.transpose() * (Rj.transpose() * Ri - Eigen::Matrix3d::Identity());
            Eigen::Matrix3d tmp_r = ric.transpose() * Rj.transpose() * Ri * ric;
            jaco_ex.rightCols<3>() = -tmp_r * Utility::skewSymmetric(pts_camera_i) + Utility::skewSymmetric(tmp_r * pts_camera_i) +
                                     Utility::skewSymmetric(ric.transpose() * (Rj.transpose() * (Ri * tic + Pi - Pj) - tic));
            jacobian_ex_pose.leftCols<6>() = reduce * jaco_ex;
            jacobian_ex_pose.rightCols<1>().setZero();
        }
        if (jacobians[3])
        {
            Eigen::Map<Eigen::Vector2d> jacobian_feature(jacobians[3]);
            jacobian_feature = reduce * ric.transpose() * Rj.transpose() * Ri * ric * pts_i_td * -1.0 / (inv_dep_i * inv_dep_i);
        }
        if (jacobians[4])
        {
            Eigen::Map<Eigen::Vector2d> jacobian_td(jacobians[4]);
            jacobian_td = reduce * ric.transpose() * Rj.transpose() * Ri * ric * velocity_i / inv_dep_i * -1.0  +
                          sqrt_info * velocity_j.head(2);
        }
    }
    sum_t += tic_toc.toc();

    return true;
}
  人工智能 最新文章
2022吴恩达机器学习课程——第二课(神经网
第十五章 规则学习
FixMatch: Simplifying Semi-Supervised Le
数据挖掘Java——Kmeans算法的实现
大脑皮层的分割方法
【翻译】GPT-3是如何工作的
论文笔记:TEACHTEXT: CrossModal Generaliz
python从零学(六)
详解Python 3.x 导入(import)
【答读者问27】backtrader不支持最新版本的
上一篇文章      下一篇文章      查看所有文章
加:2022-05-11 16:27:02  更:2022-05-11 16:29:39 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/4 15:13:52-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码