目录

  • 1.背景
  • 2.镜头成像畸变原因
  • 3.去畸变方法
  • 4. opencv去畸变函数
  • 5.代码实现


1.背景

由于相机的镜头并不完全理想,成像时会产生线条扭曲、失真等。对双目图像、鸟瞰图等进行处理时,首先要矫正去畸变。

2.镜头成像畸变原因

相机的镜头前有一块透镜,由于透镜的形状,当光线穿过透镜时,靠近光轴的光线折射比远离光轴的折射要小,就会产生径向畸变,此时真实世界中的直线在图像中会被弯曲,往外弯曲是枕形畸变,如下图(b),往里弯曲是桶形畸变,如下图(c)

图像畸变 python 图像畸变矫正opencv_图像畸变 python


由于透镜不可能完全与相机里的成像面平行,就会造成切向畸变

3.去畸变方法

假设归一化平面上的点P(x, y),可用极坐标(r, θ)表示,r就是P与归一化平面原点之间的距离,径向畸变就可以视为坐标点沿着距离方向发生改变,可以用畸变参数k1,k2,k3来修正;切向畸变可以看作坐标点的水平夹角发生改变。用多项式进行畸变拟合:

图像畸变 python 图像畸变矫正opencv_图像畸变 python_02


将这些畸变后的点在像素平面上的坐标计算出来:

图像畸变 python 图像畸变矫正opencv_系数矩阵_03


然后把畸变后的像素放回原本的位置,即去除了畸变。

4. opencv去畸变函数

void undistort( InputArray src, OutputArray dst,
                             InputArray cameraMatrix,
                             InputArray distCoeffs,
                             InputArray newCameraMatrix = noArray() );

src 输入图像
dst 输出图像
cameraMatrix 相机内参
distCoeffs 畸变系数矩阵 顺序是[k1, k2, p1, p2]

5.代码实现

int main()
{
	cv::Mat img = cv::imread("C:\\Users\\Administrator\\Desktop\\20210706221702710.png");
	cv::Mat k = cv::Mat::eye(3, 3, CV_32FC1);   // 内参矩阵
	k.at<float>(0, 0) = 458.654;
	k.at<float>(0, 1) = 0;
	k.at<float>(0, 2) = 367.215;
	k.at<float>(1, 0) = 0;
	k.at<float>(1, 1) = 457.296;
	k.at<float>(1, 2) = 248.375;
	k.at<float>(2, 2) = 1;

	cv::Mat d = cv::Mat::zeros(1, 4, CV_32FC1); // 畸变系数矩阵 顺序是[k1, k2, p1, p2]
	d.at<float>(0, 0) = -0.28340811;
	d.at<float>(0, 1) = 0.07395907;
	d.at<float>(0, 2) = 0.00019359;
	d.at<float>(0, 3) = 1.76187114e-05;

	cv::Mat img_distort, img_absdiff;
	cv::undistort(img, img_distort, k, d);
	cv::absdiff(img, img_distort, img_absdiff);
		
	cv::imshow("img", img);
	cv::imshow("img_distort", img_distort);
	cv::imshow("img_absdiff", img_absdiff);

	cv::waitKey(0);

	return 0;
}

原图:

一张工厂灰度图,由于畸变导致部分直线条明显被弯曲了

图像畸变 python 图像畸变矫正opencv_系数矩阵_04


矫正后的图:

矫正后的图像如下,可见弯曲的线都恢复正常了,但原图部分像素没有还原(落在了矫正后图像的外面)

图像畸变 python 图像畸变矫正opencv_归一化_05