使用opencv进行相机标定(python)

前言
今天介绍一个比较复杂的话题——照相机标定
为社么会复杂呢?因为解释起来比较麻烦,会涉及到很多数学公式。
本文将运用张正友相机标定的数学原理,并给出标定流程。OpenCV中已经封装好了一系列函数,我们使用这些函数可以实现张正友相机标定。那么首先说明一下什么是相机标定?
一个摄像机可以大致分为三个部分:镜头 、感光元件(CCD和CMOS)、处理电路。当光线透过镜头,会在感光元件上形成一个物体的“像”。(小孔成像原理)然后经过一系列处理就变成了电子版的图片。当然这种变换过程中会产生一定的误差(就好像透过猫眼看人会是一个曲面)。所以为了消除或者矫正这些误差,就出现了标定技术(就是将图片还原成正常人眼看到的图像)。

相机标定步骤

  1. 准备一张打印好的棋盘格,对棋盘格进行不同角度的拍摄,10-20张图片为宜;
  2. 对每张图片提取角点信息;
  3. 在图片中画出提取出的角点;
  4. 相机标定;
  5. 对标定结果评价,计算误差;
  6. 使用标定结果对原图片进行矫正;

一 、摄像机标定成像原理
1.四个坐标系
世界坐标系、相机坐标系、像素坐标系、成像平面坐标系。
我们可以把现实生活遇到的任何事物用坐标系表示出来,也可以用坐标系表示整个世界,于是便建立起了世界坐标系。想象一下,摄像机拍摄的是一张二维图片,因此整个摄像机可以用一个坐标系去标识它获取到的某个物体的位置,这是相机坐标系。像素坐标系就是相片的坐标系。成像平面坐标系类似于像素坐标系。
原本我们期望的拍摄效果:是每个坐标系中的像素都相互对应,类似于一种一元一次方程,但由于镜头或者其他关系,现在这条“直线”弯了,得到的图像也会出现“弯曲”,因此我们需要把它矫正。

2.相机参数
相机都有不同的内部参数、外部参数;

内部参数:有一个参数矩阵(fx,fy,cx,cy)和一个畸变系数(三个径向k1,k2,k3;两个切向p1,p2);内部参数是唯一的,就是一部相机只有一组内部参数。
相机将场景中的三维点变换为图像中的二维点,也就是各个坐标系变换的组合,可将变换过程整理为矩阵相乘的形式:
在这里插入图片描述在这里插入图片描述
其中,α,β表示图像上单位距离上像素的个数,则fx=αf,fy=βf

  • 6
    点赞
  • 109
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: OpenCV是一个流行的计算机视觉库,它提供了许多用于双目相机标定的函数和工具。在Python使用OpenCV进行双目相机标定,可以通过以下步骤实现: 1. 准备标定板:使用一个已知尺寸的标定板,如棋盘格,打印出来并粘贴在平面表面上。 2. 拍摄标定图像:使用双目相机拍摄多张标定图像,保证标定板在不同位置和角度下都能被看到。 3. 提取角点:使用OpenCV的函数,如cv2.findChessboardCorners(),在每张标定图像中提取标定板的角点。 4. 标定相机:使用OpenCV的函数,如cv2.calibrateCamera(),对每个相机进行单独的标定,得到相机的内参矩阵和畸变系数。 5. 计算双目相机的外参矩阵:使用OpenCV的函数,如cv2.stereoCalibrate(),对双目相机进行标定,得到相机的外参矩阵。 6. 验证标定结果:使用OpenCV的函数,如cv2.stereoRectify(),对标定结果进行验证和校正,以确保双目相机能够准确地进行深度测量。 以上就是使用OpenCV进行双目相机标定的基本步骤,需要注意的是,标定板的质量和拍摄标定图像的质量对标定结果有很大的影响,因此需要认真准备和执行每个步骤。 ### 回答2: 双目相机标定是用于确定双目相机内外参数的过程,通过双目相机标定可以获得相机的视差和深度信息,对于三维重建、立体匹配等应用非常重要。OpenCV是专门针对计算机视觉处理的一个开源库,提供了许多图像处理和计算机视觉方面的工具,其中也包括了双目相机标定的相关函数。 Python是一门高级编程语言,也是近年来被广泛应用于计算机视觉和机器学习领域的编程语言,它可以方便地调用OpenCV提供的双目相机标定函数进行标定。下面介绍一下使用Python调用OpenCV进行双目相机标定的过程。 双目相机标定需要进行以下步骤: 1.采集标定图像。需要用到一组内部、外部参数均未知的双目相机,采集至少10对以上的标定图像,最好是在不同的拍摄位置、不同的角度、不同的视角下进行拍摄。 2.提取角点。首先对采集到的每张标定图像进行角点提取,使用OpenCV中的cv2.findChessboardCorners()函数来自动检测所有角点。该函数通过输入相机标定板的规格,便可自动找到所有的内部角点,返回内部角点的二维像素坐标。 3.计算内部参数。对于每幅图像,我们需要计算其相机内部参数,使用OpenCV中的cv2.calibrateCamera()函数可以得出相机的内参矩阵、畸变系数等参数。该函数要求输入所有的标定图像的角点坐标,并计算出相机内参矩阵、外参矩阵等参数。 4.计算外部参数。计算完内部参数后,我们需要计算相机之间的外部参数,也就是相机的旋转矩阵和平移向量,使用OpenCV中的cv2.stereoCalibrate()函数可以得出相机的外参矩阵等参数。 5.计算视差图。通过双目相机得到的两幅图像,我们需要计算它们之间的视差,使用OpenCV中的cv2.StereoSGBM_create()函数可以对两幅图像进行立体匹配,并返回匹配的视差信息。 以上就是使用Python调用OpenCV进行双目相机标定的整个流程,其中可参考官方文档和相关代码示例进行学习和实践。 ### 回答3: OpenCV是一个广泛应用于计算机视觉领域的开放源代码计算机视觉库,Python使用OpenCV的一种方便的编程语言。双目相机标定是指为双目摄像机处理而设计的标定,旨在确定特定环境下使用使用视觉测量系统的误差。下面将详细介绍OpenCV使用Python进行双目相机标定的方法。 双目相机标定前需要进行以下准备: 1. 选择合适的标定板,如黑白棋盘格。 2.采集一组棋盘格图像,保持相机的标定板相对于相机光轴的姿态不变。 3.关键变量的设定,如图像尺寸和棋盘格尺寸。 接下来,我们可以使用Python实现以下步骤来完成双目相机标定: 1. 导入OpenCV库及其他必要的库: ``` import numpy as np import cv2 import glob ``` 2. 获取用于标定的图像列表。 ``` leftimage = glob.glob('左摄像机图像路径/*.jpg') rightimage = glob.glob('右摄像机图像路径/*.jpg') ``` 3. 定义标定用的棋盘格参数: ``` chessboard_size = (7, 6) square_size = 0.034 ``` 4. 创建棋盘格模型: ``` objp = np.zeros((chessboard_size[0]*chessboard_size[1],3), np.float32) objp[:,:2] = np.mgrid[0:chessboard_size[0],0:chessboard_size[1]].T.reshape(-1,2)*square_size ``` 5. 分别处理左右图像: ``` for i in range(len(leftimage)): imgL = cv2.imread(leftimage[i]) imgR = cv2.imread(rightimage[i]) grayL = cv2.cvtColor(imgL, cv2.COLOR_BGR2GRAY) grayR = cv2.cvtColor(imgR, cv2.COLOR_BGR2GRAY) retL, cornersL = cv2.findChessboardCorners(grayL, chessboard_size,None) retR, cornersR = cv2.findChessboardCorners(grayR, chessboard_size,None) if retL and retR: objpoints.append(objp) cornersL2 = cv2.cornerSubPix(grayL,cornersL,(11,11),(-1,-1),criteria) cornersR2 = cv2.cornerSubPix(grayR,cornersR,(11,11),(-1,-1),criteria) imgpointsL.append(cornersL2) imgpointsR.append(cornersR2) ``` 6. 进行标定: ``` retL, K1, D1, R1, T1 = cv2.calibrateCamera(objpoints, imgpointsL, grayL.shape[::-1], None, None) retR, K2, D2, R2, T2 = cv2.calibrateCamera(objpoints, imgpointsR, grayR.shape[::-1], None, None) ret, K1, D1, K2, D2, R, T, E, F = cv2.stereoCalibrate(objpoints, imgpointsL, imgpointsR, K1, D1, K2, D2, grayR.shape[::-1], None, None, None, None, cv2.CALIB_FIX_INTRINSIC, criteria) ``` 7. 得到最终结果: ``` R1, R2, P1, P2, Q, roi1, roi2 = cv2.stereoRectify(K1, D1, K2, D2, grayR.shape[::-1], R, T) mapLx, mapLy = cv2.initUndistortRectifyMap(K1, D1, R1, P1, grayR.shape[::-1], cv2.CV_32FC1) mapRx, mapRy = cv2.initUndistortRectifyMap(K2, D2, R2, P2, grayR.shape[::-1], cv2.CV_32FC1) ``` 通过以上代码,我们可以实现简单快捷地完成双目相机标定。除了上面介绍的步骤外,还要注意调整图像尺寸并在代码中引入必要的库和函数。在实际应用中,还有很多需要进一步优化和改进的问题,比如误差优化、噪声处理等。因此,我们需要不断学习和实践,进一步提升算法的准确性和稳定性。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值