MATLAB相机标定及OpenCV畸变校准

https://blog.csdn.net/lql0716/article/details/71973318?locationNum=8&fps=1
用相机进行拍照,从不同的角度进行拍照,为了达到较高的精度,图片的数目至少要20张左右。(注意拍照时的分辨率要固定,且图片的尺寸要一致!)

MATLAB相机标定
步骤:

启动:cameraCalibrator回车

add images

设置棋盘格的尺寸(mm),提前量一下。

设置参数。包括径向畸变参数的个数(2/3)、是否计算切向畸变。

开始标定

参数导出,保存的参数是一种特殊的数据结构,里面包含了相机内参(IntrinsicMatrix)、径向畸变(RadialDistortion)、切向畸变(TangentialDistortion)、相机的外参(RotationMatrices,TranslationVectors)等。

畸变参数,总共有五个,径向畸变3个(k1,k2,k3)和切向畸变2个(p1,p2)。
以及在OpenCV中的畸变系数的排列(这点一定要注意k1,k2,p1,p2,k3),千万不要以为k是连着的。
这里写图片描述
并且通过实验表明,三个参数的时候由于k3所对应的非线性较为剧烈。估计的不好,容易产生极大的扭曲,所以我们在MATLAB中选择使用两参数,并且选择错切和桶形畸变。
在这里插入图片描述
在这里插入图片描述
这个MATLAB函数将内矩阵和畸变矩阵输出到xml中,方便OpenCV读取。
注意事项(1.相机内参矩阵,千万记住要转置2.相机径向畸变参数向量这里是1*2的)

function writeXML(cameraParams,file)
%writeXML(cameraParams,file)
%功能:将相机校正的参数保存为xml文件
%输入:
%cameraParams:相机校正数据结构
%file:xml文件名
%说明在xml文件是由一层层的节点组成的。
%首先创建父节点 fatherNode,
%然后创建子节点 childNode=docNode.createElement(childNodeName),
%再将子节点添加到父节点 fatherNode.appendChild(childNode)
docNode = com.mathworks.xml.XMLUtils.createDocument('opencv_storage'); %创建xml文件对象
docRootNode = docNode.getDocumentElement; %获取根节点
IntrinsicMatrix = ((cameraParams.IntrinsicMatrix)'); %*相机内参矩阵,千万记住要转置*
RadialDistortion = cameraParams.RadialDistortion; %相机径向畸变参数向量1*3
TangentialDistortion =cameraParams.TangentialDistortion; %相机切向畸变向量1*2
%Distortion = [RadialDistortion(1:2),TangentialDistortion,RadialDistortion(3)]; %当RadialDistortion为3维时,构成opencv中的畸变系数向量[k1,k2,p1,p2,k3]
Distortion = [RadialDistortion(1:2),TangentialDistortion,0]; %当RadialDistortion为2维时,构成opencv中的畸变系数向量[k1,k2,p1,p2,k3]

camera_matrix = docNode.createElement('camera-matrix'); %创建mat节点
camera_matrix.setAttribute('type_id','opencv-matrix'); %设置mat节点属性
rows = docNode.createElement('rows'); %创建行节点
rows.appendChild(docNode.createTextNode(sprintf('%d',3))); %创建文本节点,并作为行的子节点
camera_matrix.appendChild(rows); %将行节点作为mat子节点

cols = docNode.createElement('cols');
cols.appendChild(docNode.createTextNode(sprintf('%d',3)));
camera_matrix.appendChild(cols);

dt = docNode.createElement('dt');
dt.appendChild(docNode.createTextNode('d'));
camera_matrix.appendChild(dt);

data = docNode.createElement('data');
for i=1:3
    for j=1:3
        data.appendChild(docNode.createTextNode(sprintf('%.16f ',IntrinsicMatrix(i,j))));
    end
    data.appendChild(docNode.createTextNode(sprintf('\n')));
end
camera_matrix.appendChild(data);
docRootNode.appendChild(camera_matrix);

distortion = docNode.createElement('distortion');
distortion.setAttribute('type_id','opencv-matrix');
rows = docNode.createElement('rows');
rows.appendChild(docNode.createTextNode(sprintf('%d',5)));
distortion.appendChild(rows);

cols = docNode.createElement('cols');
cols.appendChild(docNode.createTextNode(sprintf('%d',1)));
distortion.appendChild(cols);

dt = docNode.createElement('dt');
dt.appendChild(docNode.createTextNode('d'));
distortion.appendChild(dt);
data = docNode.createElement('data');
for i=1:5
      data.appendChild(docNode.createTextNode(sprintf('%.16f ',Distortion(i))));
end
distortion.appendChild(data);

docRootNode.appendChild(distortion);

xmlFileName = file;
xmlwrite(xmlFileName,docNode);
end

在matlab命令行输入:

writeXML(cameraParams,'cameraParams.xml');

这个OpenCV代码是将xml读取,并且对一张照片进行畸变校准。

#include <opencv2/opencv.hpp>
#include <opencv2/calib3d/calib3d.hpp>
using namespace std;
using namespace cv;

/**
 * @主函数
 */
int main( )
{

	/// 读取一副图片,不改变图片本身的颜色类型(该读取方式为DOS运行模式)
		Mat src = imread( "1.jpg");
		Mat distortion = src.clone();
		Mat camera_matrix = Mat(3, 3, CV_32FC1);
		Mat distortion_coefficients;


		//导入相机内参和畸变系数矩阵
		FileStorage file_storage("cameraParams.xml", FileStorage::READ);
		file_storage["camera-matrix"] >> camera_matrix;
		file_storage["distortion"] >> distortion_coefficients;
		file_storage.release();
		//矫正
		undistort(src, distortion, camera_matrix, distortion_coefficients);

		imshow("img", src);
		imshow("undistort", distortion);
		imwrite("undistort.jpg", distortion);

		waitKey(0);
		return 0;
}

下面是后话了。(可以不看)
利用求得的相机的内参和外参数据,可以对图像进行畸变的矫正,这里有两种方法可以达到矫正的目的,分别说明一下。

方法一:使用initUndistortRectifyMap和remap两个函数配合实现。

initUndistortRectifyMap用来计算畸变映射,remap把求得的映射应用到图像上。

initUndistortRectifyMap的函数原型:

[cpp] view plain copy 在CODE上查看代码片派生到我的代码片
//! initializes maps for cv::remap() to correct lens distortion and optionally rectify the image
CV_EXPORTS_W void initUndistortRectifyMap( InputArray cameraMatrix, InputArray distCoeffs,
InputArray R, InputArray newCameraMatrix,
Size size, int m1type, OutputArray map1, OutputArray map2 );
第一个参数cameraMatrix为之前求得的相机的内参矩阵;

第二个参数distCoeffs为之前求得的相机畸变矩阵;

第三个参数R,可选的输入,是第一和第二相机坐标之间的旋转矩阵;

第四个参数newCameraMatrix,输入的校正后的3X3摄像机矩阵;

第五个参数size,摄像机采集的无失真的图像尺寸;

第六个参数m1type,定义map1的数据类型,可以是CV_32FC1或者CV_16SC2;

第七个参数map1和第八个参数map2,输出的X/Y坐标重映射参数;

remap函数原型:

[cpp] view plain copy 在CODE上查看代码片派生到我的代码片
//! warps the image using the precomputed maps. The maps are stored in either floating-point or integer fixed-point format
CV_EXPORTS_W void remap( InputArray src, OutputArray dst,
InputArray map1, InputArray map2,
int interpolation, int borderMode=BORDER_CONSTANT,
const Scalar& borderValue=Scalar());

第一个参数src,输入参数,代表畸变的原始图像;

第二个参数dst,矫正后的输出图像,跟输入图像具有相同的类型和大小;

第三个参数map1和第四个参数map2,X坐标和Y坐标的映射;

第五个参数interpolation,定义图像的插值方式;

第六个参数borderMode,定义边界填充方式;

方法二:使用undistort函数实现

undistort函数原型:

[cpp] view plain copy 在CODE上查看代码片派生到我的代码片
//! corrects lens distortion for the given camera matrix and distortion coefficients
CV_EXPORTS_W void undistort( InputArray src, OutputArray dst,
InputArray cameraMatrix,
InputArray distCoeffs,
InputArray newCameraMatrix=noArray() );

第一个参数src,输入参数,代表畸变的原始图像;

第二个参数dst,矫正后的输出图像,跟输入图像具有相同的类型和大小;

第三个参数cameraMatrix为之前求得的相机的内参矩阵;

第四个参数distCoeffs为之前求得的相机畸变矩阵;

第五个参数newCameraMatrix,默认跟cameraMatrix保持一致;

方法一相比方法二执行效率更高一些,推荐使用。

  • 4
    点赞
  • 57
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: Matlab是一种常用的编程语言和环境,用于科学计算和工程应用。双目相机标定是一种用于精确测量和视觉定位的技术。而OpenCV是一个开源的计算机视觉库,提供了许多用于图像处理和计算机视觉的函数和工具。 要进行Matlab双目相机标定,我们可以通过访问OpenCV官网下载OpenCV库和相关教程来获得所需的功能和指导。方法如下: 1. 打开浏览器,输入 “OpenCV官网”并进入OpenCV官网。 2. 在官网主页上,找到下载链接或导航菜单中的“Downloads”(或“下载”)部分。 3. 点击下载链接,进入下载页面。 4. 在下载页面上,找到与您的操作系统和Matlab版本兼容的最新版本的OpenCV库。 5. 单击下载按钮,开始下载OpenCV库。 6. 下载完成后,将OpenCV库文件保存到您选择的目录中。 7. 解压缩下载的文件,将相关文件和文件夹复制到您的Matlab工作目录中。 8. 打开Matlab,创建并打开一个新的脚本文件。 9. 在脚本文件中,编写适当的Matlab代码来使用OpenCV库进行双目相机标定。 10. 运行脚本文件,在Matlab命令窗口或图形用户界面中查看和分析结果。 通过在Matlab中使用OpenCV库,您可以利用双目相机标定来获得相机的内部和外部参数,例如相机矩阵、畸变系数和旋转矩阵。这些参数可以用于立体视觉应用中,如深度估计、立体匹配和三维重建。 在进行双目相机标定之前,建议您先阅读相关的OpenCV教程和Matlab文档,以了解更多关于双目相机标定OpenCV库的细节和使用方法。这样能够更好地理解和应用这些技术,以获得准确且可靠的结果。 ### 回答2: 对于Matlab双目相机标定,可以通过从OpenCV官网下载相关的库和软件包来实现。OpenCV是一种开源的计算机视觉库,提供了丰富的功能和工具,包括双目相机标定。 首先,打开OpenCV官网(https://opencv.org/)并导航到下载页面。在下载页面上,你可以找到适合你操作系统的最新版本的OpenCV库和软件包。 选择合适的版本后,点击下载按钮,下载文件并保存到本地。 下载完成后,解压缩文件,找到对应的库文件和示例代码来进行Matlab双目相机标定。这个过程可能因为各操作系统的不同而有所差异。 在解压缩的文件中,找到包含示例代码和相关函数的文件夹。在这个文件夹中,你可以找到一个名为“stereo_calibrate”的示例代码或函数,用于双目相机标定。 在Matlab中打开这个示例代码或函数。根据你的需求,可以根据具体要求调整代码或函数的输入参数,如相机矩阵、畸变系数和图像坐标等。 然后,你可以使用这个示例代码或函数来读取双目相机的图像,提取图像特征,计算图像的匹配点对,并进行相机参数的标定。 最后,你可以使用标定后的相机参数来进行三维重建、深度估计或其他相关任务。 总之,通过从OpenCV官网下载相关的库和软件包,可以为Matlab提供双目相机标定的功能,方便实现双目视觉相关应用。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值