例子运行方法:
siat@hzt:~/Documents/slambook/ch7$ ./build/pose_estimation_3d3d 1.png 2.png 1_depth.png 2_depth.png
因我具体的问题是3点到3点的ICP匹配求相机姿态,所以SVD就够了,不用非线性优化(因里面有g2o库还需要再下载一个,挺大的吧,各种不想用),自己删除了此文件中非线性相关代码,并成功编译使用SVD(前提是装OPENCV,Eigen 头文件库)。新建gpose_estimation_3_3d3d.cpp内容如下:
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
using namespace std;
using namespace cv;
void pose_estimation_3d3d (
const vector& pts1,
const vector& pts2,
Mat& R, Mat& t
);
void bundleAdjustment(
const vector& points_3d,
const vector& points_2d,
Mat& R, Mat& t
);
// g2o edge
class EdgeProjectXYZRGBDPoseOnly : public g2o::BaseUnaryEdge<3, Eigen::Vector3d, g2o::VertexSE3Expmap>
{
public:
EIGEN_MAKE_ALIGNED_OPERATOR_NEW;
EdgeProjectXYZRGBDPoseOnly( const Eigen::Vector3d& point ) : _point(point) {}
virtual void computeError()
{
const g2o::VertexSE3Expmap* pose = static_cast ( _vertices[0] );
// measurement is p, point is p'
_error = _measurement - pose->estimate().map( _point );
}
virtual void linearizeOplus()
{
g2o::VertexSE3Expmap* pose = static_cast(_vertices[0]);
g2o::SE3Quat T(pose->estimate());
Eigen::Vector3d xyz_trans = T.map(_point);
double x = xyz_trans[0];
double y = xyz_trans[1];
double z = xyz_trans[2];
_jacobianOplusXi(0,0) = 0;
_jacobianOplusXi(0,1) = -z;
_jacobianOplusXi(0,2) = y;
_jacobianOplusXi(0,3) = -1;
_jacobianOplusXi(0,4) = 0;
_jacobianOplusXi(0,5) = 0;
_jacobianOplusXi(1,0) = z;
_jacobianOplusXi(1,1) = 0;
_jacobianOplusXi(1,2) = -x;
_jacobianOplusXi(1,3) = 0;
_jacobianOplusXi(1,4) = -1;
_jacobianOplusXi(1,5) = 0;
_jacobianOplusXi(2,0) = -y;
_jacobianOplusXi(2,1) = x;
_jacobianOplusXi(2,2) = 0;
_jacobianOplusXi(2,3) = 0;
_jacobianOplusXi(2,4) = 0;
_jacobianOplusXi(2,5) = -1;
}
bool read ( istream& in ) {}
bool write ( ostream& out ) const {}
protected:
Eigen::Vector3d _point;
};
int main ( int argc, char** argv )
{
vector pts1, pts2;
float x1=43,y1=7,z1=404; } void pose_estimation_3d3d ( } void bundleAdjustment ( }
float x2=-38,y2=6,z2=384;
float x3=-52,y3=2,z3=382;
float xo1=78,yo1=6,zo1=416;
float xo2=6,yo2=7,zo2=372;
float xo3=-7,yo3=2,zo3=367;
pts1.push_back ( Point3f (x1, y1, z1));
pts2.push_back ( Point3f (xo1, yo1,zo1));
pts1.push_back ( Point3f (x2, y2, z2));
pts2.push_back ( Point3f (xo2, yo2,zo2));
pts1.push_back ( Point3f (x3, y3, z3));
pts2.push_back ( Point3f (xo3, yo3,zo3));
cout<<"3d-3d pairs: "<
pose_estimation_3d3d ( pts1, pts2, R, t );
cout<<"ICP via SVD results: "<cout<<"calling bundle adjustment"<
const vector& pts1,
const vector& pts2,
Mat& R, Mat& t
)
{
Point3f p1, p2; // center of mass
int N = pts1.size();
for ( int i=0; i
p1 += pts1[i];
p2 += pts2[i];
}
p1 = Point3f( Vec3f(p1) / N);
p2 = Point3f( Vec3f(p2) / N);
vector q1 ( N ), q2 ( N ); // remove the center
for ( int i=0; i
q1[i] = pts1[i] - p1;
q2[i] = pts2[i] - p2;
}// compute q1*q2^T
Eigen::Matrix3d W = Eigen::Matrix3d::Zero();
for ( int i=0; i
const vector< Point3f >& pts1,
const vector< Point3f >& pts2,
Mat& R, Mat& t )
{
// 初始化g2o
typedef g2o::BlockSolver< g2o::BlockSolverTraits<6,3> > Block; // pose维度为 6, landmark 维度为 3
Block::LinearSolverType* linearSolver = new g2o::LinearSolverEigenBlock::PoseMatrixType(); // 线性方程求解器
Block* solver_ptr = new Block( linearSolver ); // 矩阵块求解器
g2o::OptimizationAlgorithmGaussNewton* solver = new g2o::OptimizationAlgorithmGaussNewton( solver_ptr );
g2o::SparseOptimizer optimizer;
optimizer.setAlgorithm( solver );// vertex
g2o::VertexSE3Expmap* pose = new g2o::VertexSE3Expmap(); // camera pose
pose->setId(0);
pose->setEstimate( g2o::SE3Quat(
Eigen::Matrix3d::Identity(),
Eigen::Vector3d( 0,0,0 )
) );
optimizer.addVertex( pose );
// edges
int index = 1;
vector
然后在同一级的CMakeLists.txt文件中(末尾)加入一段编译信息:
add_executable( pose_estimation_3_3d3d pose_estimation_3_3d3d.cpp )
target_link_libraries( pose_estimation_3_3d3d
${OpenCV_LIBS}
g2o_core g2o_stuff g2o_types_sba g2o_csparse_extension
C S P A R S E L I B R A R Y ) 然 后 进 入 b u i l d 目 录 开 始 编 译 : s i a t @ h z t : / D o c u m e n t s / s l a m b o o k / c h 7 {CSPARSE_LIBRARY} ) 然后进入build目录开始编译: siat@hzt:~/Documents/slambook/ch7 CSPARSELIBRARY)然后进入build目录开始编译:siat@hzt: /Documents/slambook/ch7 cd build
siat@hzt:~/Documents/slambook/ch7/build$ cmake …(注意…前有一个空格)
siat@hzt:~/Documents/slambook/ch7/build$ cmake…
过几分钟编译完生产可执行文件pose_estimation_3_3d3d,
就可以运行了:
siat@hzt:~/Documents/slambook/ch7/build$ ./pose_estimation_3_3d3d