草庐IT

ceres 求slam 或者 SfM的协方差及ceres 四元数求导

cheng.li@3D_Vision 2023-04-19 原文

ceres 如何进行后验估计

ceres 求landmark的协方差矩阵

以下代码是自己在colmap 中实现的

/// 计算每个3D points的协方差
 for (const image_t image_id :config_.Images()) {
   Image& image = reconstruction->Image(image_id);
   for (const Point2D& point2D : image.Points2D()) {
     if (!point2D.HasPoint3D()) {
       continue;
     }
       Point3D& point3D = reconstruction->Point3D(point2D.Point3DId());
       Eigen::Matrix<double,3,3,Eigen::RowMajor> cov_tvec =Eigen::Matrix<double,3,3,Eigen::RowMajor>::Zero();
       ceres::Covariance::Options cov_options;
       ceres::Covariance covariance(cov_options);
       std::vector<std::pair<const double *,const double *> >covariance_blocks;
//       covariance_blocks.push_back(std::make_pair(image.Tvec().data(),image.Tvec().data()));
       covariance_blocks.push_back(std::make_pair(point3D.XYZ().data(),point3D.XYZ().data()));
       covariance.Compute(covariance_blocks,problem_.get());
     // 在切空间get 协方差矩阵
       covariance.GetCovarianceBlockInTangentSpace(point3D.XYZ().data(),point3D.XYZ().data(),cov_tvec.data());
       std::cout<<"image_id:"<<image_id<<" point3D_id: "<<point2D.Point3DId()<<" covariance:\n"<<cov_tvec<<"\n";
   }
 }

ceres 四元数求导

  • 关于manifold 和tangent space
    许多优化问题,特别是传感器融合算法,都是在manifold 空间建模,如以四元数为代表的方向传感器.但是在优化 Manifold 上的变量时需要注意,Manifold 上变量是 over parameterized,即 Manifold 上变量的维度大于其自由度。这就造成变量建立的误差方之间程存在约束,如果直接求导会很困难,故要转换在切空间求导.
    ceres LocalParameterization 上原话如下:

Working in the tangent space reduces not just the computational complexity of the optimization algorithm, but also improves the numerical behaviour of the algorithm.

  • 对四元数求导的代码如下:
// ceres local_parameterization.cc 源码
bool QuaternionParameterization::Plus(const double* x,
                                      const double* delta,
                                      double* x_plus_delta) const {
  const double norm_delta =
      sqrt(delta[0] * delta[0] + delta[1] * delta[1] + delta[2] * delta[2]);
  if (norm_delta > 0.0) {
    const double sin_delta_by_delta = (sin(norm_delta) / norm_delta);
    double q_delta[4];
    q_delta[0] = cos(norm_delta);
    q_delta[1] = sin_delta_by_delta * delta[0];
    q_delta[2] = sin_delta_by_delta * delta[1];
    q_delta[3] = sin_delta_by_delta * delta[2];
    QuaternionProduct(q_delta, x, x_plus_delta);
  } else {
    for (int i = 0; i < 4; ++i) {
      x_plus_delta[i] = x[i];
    }
  }
  return true;
}
bool QuaternionParameterization::ComputeJacobian(const double* x,
                                                 double* jacobian) const {
  jacobian[0] = -x[1]; jacobian[1]  = -x[2]; jacobian[2]  = -x[3];  // NOLINT
  jacobian[3] =  x[0]; jacobian[4]  =  x[3]; jacobian[5]  = -x[2];  // NOLINT
  jacobian[6] = -x[3]; jacobian[7]  =  x[0]; jacobian[8]  =  x[1];  // NOLINT
  jacobian[9] =  x[2]; jacobian[10] = -x[1]; jacobian[11] =  x[0];  // NOLINT
  return true;
}
//////轴角到四元数,可以看到QuaternionParameterization::Plus操作的半轴角,需要注意
template<typename T>
inline void AngleAxisToQuaternion(const T* angle_axis, T* quaternion) {
  const T& a0 = angle_axis[0];
  const T& a1 = angle_axis[1];
  const T& a2 = angle_axis[2];
  const T theta_squared = a0 * a0 + a1 * a1 + a2 * a2;

  // For points not at the origin, the full conversion is numerically stable.
  if (theta_squared > T(0.0)) {
    const T theta = sqrt(theta_squared);
    const T half_theta = theta * T(0.5);
    const T k = sin(half_theta) / theta;
    quaternion[0] = cos(half_theta);
    quaternion[1] = a0 * k;
    quaternion[2] = a1 * k;
    quaternion[3] = a2 * k;
  } else {
    // At the origin, sqrt() will produce NaN in the derivative since
    // the argument is zero.  By approximating with a Taylor series,
    // and truncating at one term, the value and first derivatives will be
    // computed correctly when Jets are used.
    const T k(0.5);
    quaternion[0] = T(1.0);
    quaternion[1] = a0 * k;
    quaternion[2] = a1 * k;
    quaternion[3] = a2 * k;
  }
}

有关ceres 求slam 或者 SfM的协方差及ceres 四元数求导的更多相关文章

  1. 欧拉角、旋转矩阵及四元数 - 2

    欧拉角、旋转矩阵及四元数1.简介2.欧拉角2.1欧拉角定义2.2右手系和左手系2.3转换流程3.旋转矩阵4.四元数4.1四元数与欧拉角和旋转矩阵之间等效变换4.2测试Matlab代码5.总结1.简介常用姿态参数表达方式包括方向余弦矩阵、欧拉轴/角参数、欧拉角、四元数以及罗德里格参数等。高分辨率光学遥感卫星主要采用欧拉角与四元数对姿态参数进行描述。这里着重讲解欧拉角、旋转矩阵和四元数。2.欧拉角2.1欧拉角定义欧拉角是表征刚体旋转的一种方法之一,由莱昂哈德·欧拉引入的三个角度,用于描述刚体相对于固定坐标系的方向。在摄影测量、空间科学或其它技术领域,一般用一组(三个)欧拉角描述两个空间坐标之间的旋

  2. ruby - 我可以在传递给方法的 block 上强制执行元数吗? - 2

    有什么方法可以“开启”使用Proc.new或Kernel.proc实例化的Proc的严格元数强制执行,使其表现得像Proc用lambda实例化?我的initialize方法采用block&action并将其分配给实例变量。我希望action严格执行arity,因此当我稍后对其应用参数时,它会引发一个ArgumentError,我可以挽救它并引发一个更有意义的异常。基本上:classCommandattr_reader:name,:actiondefinitialize(name,&action)@name=name@action=actionenddefperform(*args)be

  3. ruby - 应该 validate_format_of 。 not_with 在框架中有问题(或者在我的理解中) - 2

    我将以下代码放入RSpec测试中:it{shouldvalidate_format_of(:email).not_with('test@test')}并设置实际的类:validates:email,:presence=>true,:format=>/\b[A-Z0-9._%-]+@(?:[A-Z0-9-]+\.)+[A-Z]{2,4}\b/i当我运行测试时,我得到:失败:1)用户失败/错误:它{应该validate_format_of(:email).not_with('test@test')}当电子邮件设置为“test@test”时,预期错误包括“can'tbeblank”,得到错误

  4. ruby - 如何获取 "super"的元数? - 2

    假设您正在用不同的元数覆盖子类中的方法:classAdeffoo(arg)#arityis1#doingsomethinghereendendclassB有没有办法在线获取super的元数?(实际用例:我调用super时知道父类(superclass)不接受任何参数。但是,如果父类(superclass)实现(在gem中)发生变化,我想发出警告。)感谢您的帮助! 最佳答案 关于您的实际用例:无需亲自检查参数。打电话super(arg1)如果参数计数不匹配,Ruby将引发ArgumentError。更新由于一些反对票,我想我应该回答你

  5. Ruby 等同于 C#'s ' yield' 关键字,或者,在不预分配内存的情况下创建序列 - 2

    在C#中,您可以这样做:publicIEnumerableGetItems(){for(inti=0;i这将返回一个包含1000万个整数的可枚举序列,而无需在该长度的内存中分配一个集合。有没有一种方法可以在Ruby中做同样的事情?我要处理的具体示例是将矩形数组展平为要枚举的值序列。返回值不必是Array或Set,而是某种只能按顺序而不是索引迭代/枚举的序列。因此,整个序列不需要同时分配到内存中。在.NET中,这是IEnumerable和IEnumerable.对Ruby世界中此处使用的术语的任何澄清都会有所帮助,因为我更熟悉.NET术语。编辑也许我最初的问题还不够清楚——我认为yiel

  6. ruby-on-rails - 你可以用ruby重新定义一个类吗?或者这只是在 irb - 2

    我启动了irb,然后输入:类点结束然后我再次输入,但添加了一些其他内容。Irb没有提示我正在定义一个已经存在的类。 最佳答案 其实你并没有重新定义Point类,你重新打开了它。一个小代码片段来说明差异:classPointdeffooendendclassPointdefbarendend现在Point有两个方法:foo和bar。所以Point的第二个定义并没有取代之前的定义,而是添加了它。这在ruby​​脚本和irb中都是可能的(标准库中的类也是可能的,而不仅仅是您自己的类)。也可以真正重新定义类,通过使用remove_const

  7. Ruby:如何将变量设置为 0,或者如果已经设置,则递增 1 - 2

    我知道||=运算符,但我认为它不会对我有帮助...尝试创建一个数组来计算对象数组中“类型”的数量。array.eachdo|c|newarray[c.type]=newarray[c.type]?newarray[c.type]+1?0end有没有更优雅的方式来做到这一点? 最佳答案 types=Hash.new(-1)#Itfeelslikethisshouldbe0,buttobe#equivalenttoyourexampleitneedstobe-1array.eachdo|c|types[c.type]+=1end

  8. 【Chano的SFM教程】3dmax 面部表情.VTA基本制作教程 - 2

    本篇教程作者为:小鸟Chano,转载请表明作者和出处:CSDN欢迎观看本次教程本教程将会为你演示使用3DMAX制作一个基本的SFM表情控制器【表情滑条】并导入SFM进行使用。Chano自己也是近期才掌握的这项知识,所以过程中可能有很多迷之操作和瑕疵还请见谅哈^^~1、操作过程首先,请转到wunderboy网站。获取我们需要的插件,下载并安装**“3DSMaxVTAexportplug-in”**(1)如何安装?只需将其放入plugins文件夹,你也可以在这里找到其他相关插件,比如SMD导入导出、VTF插件。温馨提示:本教程不包含任何有关MAX的基本操作知识。本教程全程没有语音讲解,如条件不允许

  9. ruby - 将元数据添加到 PDF - 2

    我需要将元数据添加到我使用prawn创建的PDF中.该元数据稍后可能会被pdf-reader提取。.此元数据将包含内部文档编号和下游工具所需的其他信息。将元数据与PDF的每一页相关联会很方便。ThePDFspecification声称我可以将每页私有(private)数据存储在“Page-PieceDictionary”中。第14.5节指出:Apage-piecedictionary(PDF1.3)maybeusedtoholdprivateconformingproductdata.ThedatamaybeassociatedwithapageorformXObjectbymeans

  10. ruby-on-rails - ruby::Module 或者只是 Module - 2

    我正在慢慢浏览Rails源代码,以便更好地掌握ruby​​和Rails。在以下rails类中test_case.rb这条线是classTestCase我想知道执行以下操作是否有任何区别classTestCase这可能看起来微不足道,但这些事情对于学习一门新语言来说很重要。如果我删除前导::,测试仍然针对ActiveSupport运行,那么它做了什么......:P 最佳答案 ::Test确保您获得名为Test的顶层模块。后一种情况(Test::Unit::TestCase)不能确保Test是顶层模块,例如,它可能是一个类。这意味着它

随机推荐