*对极几何 求解 两组单目相机 2D图像 间的 旋转平移 矩阵
* 2D 点对 求 两相机的 旋转和平移矩阵
* 空间点 P 两相机 像素点对 p1 p2 两相机 归一化平面上的点对 x1 x2 与P点对应
* 相机内参数 K 两镜头旋转平移矩阵 R t 或者 变换矩阵 T
* p1 = KP (世界坐标系) p2 = K( RP + t) = KTP
* 而 x1 = K逆* p1 x2 = K逆* p2 相机坐标系下 归一化平面上的点 x1= (px -cx)/fx x2= (py -cy)/fy
* 所以 x1 = P x2 = R x1 + t
* t 外积 x2 = t 外积 R x1 ;t 外积 t =0 sin(cet) =0 垂线段投影
* x2转置 * t 外积 x2 = x2转置 * t 外积 R x1 = 0
* 有 x2转置 * t 外积 R x1 = x2转置 * E * x1 = 0 ; E 为本质矩阵
* p2转置 * K 转置逆 * t 外积 R * K逆 * p1 =p2转置 * F * p1 = 0 ;
* F 为基础矩阵
* x2转置 * E * x1 = 0 x1 x2 为 由 像素坐标转化的归一化坐标
* 一个点对一个约束 ,8点法 可以算出 E的各个元素 ,
* 再 奇异值分解 E 得到 R t
* 单应矩阵描述了两个平面间的映射关系
* p2 = K( RP + t) 有 P在极平面上 平面方程 n转置 * P + d = 0
* 得到 - n转置 * P/d =1
* p2 = K( RP - t *n转置 * P/d) = K( R - t *n转置 * /d)*P = H *p1
* p2 = H *p1
* 一个点对 2个约束
* 4 点法求解 单应矩阵 H 再对 H进行分解

github 连接 点击打开链接

#include <iostream>
#include <opencv2/core/core.hpp>
#include <opencv2/features2d/features2d.hpp>
#include <opencv2/highgui/highgui.hpp>
#include <opencv2/calib3d/calib3d.hpp>
// #include "extra.h" // use this if in OpenCV2
using namespace std;//标准库 命名空间
using namespace cv; //opencv库命名空间// ./pose_estimation_2d2d 1.png 2.png
/***************************************************** 本程序演示了如何使用2D-2D的特征匹配估计相机运动  * **************************************************/
//特征匹配 计算匹配点对
void find_feature_matches (const Mat& img_1, const Mat& img_2, // & 为引用  直接使用 参数本身 不进行复制  节省时间std::vector<KeyPoint>& keypoints_1,// std::vector<KeyPoint>& keypoints_2,std::vector< DMatch >& matches );//keypoint Descriptors Match   描述子匹配
//位置估计 计算旋转和平移
void pose_estimation_2d2d (std::vector<KeyPoint> keypoints_1,std::vector<KeyPoint> keypoints_2,std::vector< DMatch > matches,Mat& R, Mat& t );// 像素坐标转相机归一化坐标
Point2d pixel2cam ( const Point2d& p, const Mat& K );int main ( int argc, char** argv )
{if ( argc != 3 )//命令行参数  1.png  2.png{cout<<"usage: pose_estimation_2d2d img1 img2"<<endl;return 1;}//-- 读取图像Mat img_1 = imread ( argv[1], CV_LOAD_IMAGE_COLOR );//彩色图模式Mat img_2 = imread ( argv[2], CV_LOAD_IMAGE_COLOR );vector<KeyPoint> keypoints_1, keypoints_2;//关键点vector<DMatch> matches;//特征点匹配对find_feature_matches ( img_1, img_2, keypoints_1, keypoints_2, matches );//得到匹配点对cout<<"一共找到了"<<matches.size() <<"组匹配点"<<endl;//-- 估计两张图像间运动Mat R,t;//旋转和平移 第一张图 到第二章图的坐标变换矩阵和平移矩阵pose_estimation_2d2d ( keypoints_1, keypoints_2, matches, R, t );//-- 验证E=t^R*scale    变叉乘 为 矩阵乘法   三阶行列式     // https://www.cnblogs.com/monoSLAM/p/5349497.html//  a   b   c //  d  e    f     = bf - ce , cd - af , ae -bd  =  [0 - c  b; c 0 -a; -b a 0 ] * [ d e f]// 向量 t = [ a1 a2 a3] 其//  叉乘矩阵 = [0  -a3  a2;//                      a3  0  -a1; //                     -a2 a1  0 ]  Mat t_x = ( Mat_<double> ( 3,3 ) << //t向量的 叉乘矩阵0,                      -t.at<double> ( 2,0 ),     t.at<double> ( 1,0 ),t.at<double> ( 2,0 ),      0,                      -t.at<double> ( 0,0 ),-t.at<double> ( 1.0 ),     t.at<double> ( 0,0 ),      0 );cout<<"本质矩阵E = t^R= t叉乘矩阵 * R = "<<endl<<t_x*R<<endl;//-- 验证对极约束//相机内参数//   [fx 0 cx//     0 fy cy//     0 0  1]Mat K = ( Mat_<double> ( 3,3 ) << 520.9, 0, 325.1, 0, 521.0, 249.7, 0, 0, 1 );// 相机内参,TUM Freiburg2for ( DMatch m: matches )//Descriptors Match   描述子匹配{Point2d pt1 = pixel2cam ( keypoints_1[ m.queryIdx ].pt, K );//像素坐标转相机归一化坐标    x1 =  K逆* p1  x2 =  K逆* p2  相机坐标系下 归一化平面上的点 Mat y1 = ( Mat_<double> ( 3,1 ) << pt1.x, pt1.y, 1 );// 归一化平面上的点 齐次表示Point2d pt2 = pixel2cam ( keypoints_2[ m.trainIdx ].pt, K );Mat y2 = ( Mat_<double> ( 3,1 ) << pt2.x, pt2.y, 1 );Mat d = y2.t() * t_x * R * y1;//理论上为 0 cout << "epipolar constraint = " << d << endl;}return 0;
}//特征匹配 计算匹配点对 函数
void find_feature_matches ( const Mat& img_1, const Mat& img_2,std::vector<KeyPoint>& keypoints_1,std::vector<KeyPoint>& keypoints_2,std::vector< DMatch >& matches )
{//--------------------第0部初始化------------------------------------------------------Mat descriptors_1, descriptors_2;//描述子//  OpenCV3 特征点检测器  描述子生成器 用法Ptr<FeatureDetector> detector = ORB::create();         //特征点检测器    其他 BRISK   FREAK   Ptr<DescriptorExtractor> descriptor = ORB::create();//描述子生成器// OpenCV2 特征点检测器  描述子生成器 用法// Ptr<FeatureDetector> detector = FeatureDetector::create ( "ORB" );// Ptr<DescriptorExtractor> descriptor = DescriptorExtractor::create ( "ORB" );Ptr<DescriptorMatcher> matcher  = DescriptorMatcher::create ( "BruteForce-Hamming" );//二进制描述子 汉明点对匹配//------------------第一步:检测 Oriented FAST 角点位置-----------------------------detector->detect ( img_1,keypoints_1 );detector->detect ( img_2,keypoints_2 );//------------------第二步:根据角点位置计算 BRIEF 描述子-------------------------descriptor->compute ( img_1, keypoints_1, descriptors_1 );descriptor->compute ( img_2, keypoints_2, descriptors_2 );//------------------第三步:对两幅图像中的BRIEF描述子进行匹配,使用 Hamming 距离vector<DMatch> match;//Descriptors Match   描述子匹配//BFMatcher matcher ( NORM_HAMMING );matcher->match ( descriptors_1, descriptors_2, match );//各个特征点描述子之间的汉明距离匹配//-----------------第四步:匹配点对筛选--------------------------------------------------double min_dist=10000, max_dist=0;//找出所有匹配之间的最小距离和最大距离, 即是最相似的和最不相似的两组点之间的距离for ( int i = 0; i < descriptors_1.rows; i++ ){double dist = matches[i].distance;if ( dist < min_dist ) min_dist = dist;  //最短距离  最相似if ( dist > max_dist ) max_dist = dist; //最长距离 最不相似}printf ( "-- Max dist : %f \n", max_dist );printf ( "-- Min dist : %f \n", min_dist );//当描述子之间的距离大于两倍的最小距离时,即认为匹配有误.但有时候最小距离会非常小,设置一个经验值30作为下限.for ( int i = 0; i < descriptors_1.rows; i++ ){if ( match[i].distance <= max ( 2*min_dist, 30.0 ) )//最大距离{matches.push_back ( match[i] );}}
}// 像素坐标转相机归一化坐标
// 像素坐标转相机归一化坐标    x1 =  K逆* p1  x2 =  K逆* p2  相机坐标系下 归一化平面上的点
Point2d pixel2cam ( const Point2d& p, const Mat& K )
{return Point2d(( p.x - K.at<double> ( 0,2 ) ) / K.at<double> ( 0,0 ),// x= (px -cx)/fx( p.y - K.at<double> ( 1,2 ) ) / K.at<double> ( 1,1 )//  y=(py-cy)/fy);
}//特征匹配 计算匹配点对 函数   第一张图 到第二章图的坐标变换矩阵和平移矩阵
//对极几何
void pose_estimation_2d2d ( std::vector<KeyPoint> keypoints_1,std::vector<KeyPoint> keypoints_2,std::vector< DMatch > matches,Mat& R, Mat& t )
{// 相机内参,TUM Freiburg2//相机内参数//   [fx 0 cx//     0 fy cy//     0 0  1]Mat K = ( Mat_<double> ( 3,3 ) << 520.9, 0, 325.1, 0, 521.0, 249.7, 0, 0, 1 );//------------把匹配点转换为vector<Point2f>的形式------------------vector<Point2f> points1;vector<Point2f> points2;for ( int i = 0; i < ( int ) matches.size(); i++ ){points1.push_back ( keypoints_1[matches[i].queryIdx].pt );points2.push_back ( keypoints_2[matches[i].trainIdx].pt );}//-----------计算基础矩阵 F    p2转置*F*p1 = 0   -----------------------------------------------------Mat fundamental_matrix;fundamental_matrix = findFundamentalMat ( points1, points2, CV_FM_8POINT );//8点发    p2转置*F*p1 = 0 cout<<"基础矩阵 fundamental_matrix is "<<endl<< fundamental_matrix<<endl;//-----------计算本质矩阵 E   x2转置 * E * x1 = 0  ----------------------------------------------------Point2d principal_point ( 325.1, 249.7 );    //相机光心, TUM dataset标定值   cx    cydouble focal_length = 521;            //相机焦距, TUM dataset标定值  fx     fyMat essential_matrix;essential_matrix = findEssentialMat ( points1, points2, focal_length, principal_point );// x2转置 * E * x1 = 0  cout<<"本质矩阵 essential_matrix is "<<endl<< essential_matrix<<endl;//-----------计算单应矩阵H    p2 = H *p1  ---------------------------------------------------Mat homography_matrix;homography_matrix = findHomography ( points1, points2, RANSAC, 3 );cout<<"单应矩阵 homography_matrix is "<<endl<<homography_matrix<<endl;//-- 从本质矩阵中恢复旋转和平移信息. 使用奇异值分解法得到recoverPose ( essential_matrix, points1, points2, R, t, focal_length, principal_point );// E = t^R = U C  V   ,U   V 为正交矩阵   C 为奇异值矩阵 C =  diag(1, 1, 0)cout<<"R is "<<endl<<R<<endl;cout<<"t is "<<endl<<t<<endl;}

2D - 2D 点对 求解基础矩阵 F 本质矩阵E 单应矩阵 H 进而求 旋转矩阵 R 和 t相关推荐

  1. linux vim编辑矩阵乘,LM算法计算单应矩阵

    LM算法计算单应矩阵 slam 单应矩阵 单应矩阵的定义 什么是单应矩阵呢?其实简单来说,就是两个图像之间的变换矩阵.什么意思呢,可以考虑这样一个情形: 你有一个相机,拍摄一个建筑物,首先在某一个视角 ...

  2. 多视图几何总结——单应矩阵和基础矩阵的兼容关系

    多视图几何总结--单应矩阵和基础矩阵的兼容关系 多视图几何总结--单应矩阵和基础矩阵的兼容关系 (1)单应矩阵和基础矩阵的兼容性 (2)基础矩阵 -> 单应矩阵 (3)单应矩阵 -> 基础 ...

  3. 通过SVD求解单应矩阵

    我们现在知道原则上4对匹配点对就可以唯一确定单应矩阵,但是在实际应用中我们无法保证两个视图严格满足使用条件(只有旋转变换:远景:平面场景),所以要使用拟合的方法求一个最优解.现在就来以SIFT算法源码 ...

  4. 09_对极约束与单应矩阵

    前言 根据上一节的介绍,我们已经有了匹配好的点对,然后我们就可以根据点对估计相机的运动.这里由于相机的原理不同,我们要分情况讨论: 如果是单目相机,此时我们仅知道2D的像素坐标,要根据两组2D点估计相 ...

  5. 什么是单应矩阵和本质矩阵

    知乎上面的大牛还是很多,直接搜Homography或者单应矩阵就能得到很多大神的回答,可能回答中的一句话或者一个链接就够自己学习很久. 其实在之前研究双目视觉的时候就接触了对极几何,通过视觉就可以得到 ...

  6. 基本矩阵、本质矩阵和单应矩阵

    两幅视图存在两个关系:第一种,通过对极几何一幅图像上的点可以确定另外一幅图像上的一条直线:另外一种,通过上一种映射,一幅图像上的点可以确定另外一幅图像上的一个点,这个点是第一幅图像通过光心和图像点的射 ...

  7. MVG读书笔记——单应矩阵估计这件小事(一)

    参数估计是计算机视觉中经常遇到的一个问题,为较好的估计参数,人们发明了各种各样的算法.这里我们就以单应矩阵H的估计为例,一个个介绍这些常用算法. DLT算法 DLT(direct linear tra ...

  8. Opencv学习(3)——基础矩阵F、本质矩阵E、单应矩阵H 函数解析

    官网:https://docs.opencv.org/3.4.0/d9/d0c/group__calib3d.html#ga4abc2ece9fab9398f2e560d53c8c9780 基础矩阵F ...

  9. 多视图几何总结——基础矩阵、本质矩阵和单应矩阵的求解过程

    多视图几何总结--基础矩阵.本质矩阵和单应矩阵的求解过程 多视图几何总结--基础矩阵.本质矩阵和单应矩阵的求解过程 1. 说明--其实求解过程大同小异 2. 单应矩阵求解过程 2.1 基于代数误差的线 ...

最新文章

  1. java集合中某一个元素出现的次数
  2. vsftp匿名访问目录_vsftp 匿名访问设置设置
  3. awk,sed,cut获取最后一列数据
  4. python代码大全p-代码这样写更优雅(Python版)
  5. javascript编程风格
  6. Spark Structured Straming:'writeStream' can be called only on streaming Dataset/DataFrame
  7. 【动态规划笔记】状压dp:蓝桥 矩阵计数 (二进制枚举)
  8. 轻量级的实现复制文本到剪贴板功能的 js
  9. SVN中,A项目如何共享B项目的内容
  10. amd 安卓虚拟机_一款完美解决AMD兼容问题的安卓模拟器
  11. 朱加强计算机网络技术,深入对比三种标准拓扑结构
  12. 三星手机双微信设置方法,三星手机如何双开微信
  13. 3533: 黑白图像
  14. 实例——领域驱动设计DDD
  15. 蘑菇街2016校园招聘之编程题解析-技术类
  16. matlab分组形式条形图,Matplotlib带标签的分组条形图
  17. python如何下载安装spacy_SpaCy下载及安装
  18. 争对让望对思野葛对山栀注解_解析
  19. 基于OpenWRT的软件开发流程
  20. Swing学习----------QQ登录界面制作(一)

热门文章

  1. 2018.8.4T2(贪心,dp,线段树,优先队列)
  2. euler欧拉系统尝试用yum、源码安装pacemaker失败
  3. Linux中tty、pty、pts的概念及区别
  4. 收购家乐福中国,苏宁要做新零售第三极?
  5. 乱杀HTML知识点(小白版本)
  6. 情人节的表白爱心来了
  7. JavaScript的继承,原型和原型链
  8. 解决安卓应用程序未安装的三种方法
  9. Etyma01 ced ceed cess
  10. 通信原理眼图画法_四川大学通信原理眼图实验