所以基本上我想找出,对于我在这里所做的校准,反转失真参数是什么。
注意:我知道我们可以执行不失真,然后使用重Map来完成我在下面所做的工作,但我的目标是能够找出实际的反转失真参数,并使用它们来扭曲其他图像,而不仅仅是能够还原cv2.undistort()所做的事
概述:
我尝试过传递失真参数的否定:
# _, mat, distortion, _, _ = cv2.calibrateCamera(...)
# undistorted_image = cv2.undistort(with distortion)
# redistorted_image = cv2.undistort(with np.negative(distortion))
理论上,我在想,如果重新扭曲的图像与原始图像相似,那么 np.negative(distortion)
参数是我要找的,但结果是假的。
我使用的实际方法:
def save_undistorted_images(image, matrix, distortion):
test_image_su = cv.imread(image)
height, width = test_image_su.shape[:2]
new_camera_mtx, roi = cv.getOptimalNewCameraMatrix(matrix, distortion, (width, height), 1, (width, height))
distortion_u = np.negative(distortion)
# unsure if this line helps
new_camera_mtx_inv, roi = cv.getOptimalNewCameraMatrix(matrix, distortion_u, (width, height), 1, (width, height))
# undistort the image
undistorted_image = cv.undistort(test_image_su, matrix, distortion, None, new_camera_mtx)
cv.imwrite('undistorted_frame.png', undistorted_image)
# redistort trying to get something like original image (image_test_su)
distorted_image = cv.undistort(undistorted_image, matrix, distortion_u, None, new_camera_mtx_inv)
cv.imwrite('redistorted_frame.png', distorted_image)
结果是:
(左a:原始)(右b:未失真)
(左c:使用 np.negative(distortion)
)(右d:使用 np.negative(distortion)
))
这里的图像d基本上是c在b上执行的,我期望它类似于a
为什么这里的b会压倒c的作用?
我尝试过的另一种计算逆的方法:
下面是我对本文的python实现
distortion_u = distortion
k1 = distortion_u[0][0]
k2 = distortion_u[0][1]
k3 = distortion_u[0][4]
b1 = -1 * k1
b2 = 3 * (k1 * k1) - k2
b3 = (8 * k1 * k2) + (-1 * (12 * (k1 * k1 * k1))) - k3
# radial:
distortion_u[0][0] = b1
distortion_u[0][1] = b2
distortion_u[0][4] = b3
# tangential:
#distortion_u[0][2] = -1 * distortion_u[0][2]
#distortion_u[0][3] = -1 * distortion_u[0][3]
利用上述畸变参数对未畸变图像进行畸变处理的效果也不好,看起来与上述结果非常相似。
所以,这让我们想到:
为什么正态失真的影响总是压倒负的np(失真)或其他什么?
所有的扭曲都是这样的吗(负值不等于相反的效果)
如何得到实际相反的畸变参数?
1条答案
按热度按时间7vhp5slm1#
恐怕你做错了。已校准的opencv失真模型从失真的图像坐标计算未失真和标准化的图像坐标。它是一个非线性模型,所以反演它需要求解一个非线性(多项式)方程组。
仅在单参数纯径向畸变的情况下,即当唯一的非零畸变参数为k1时,r^2的系数才存在闭式(参数)解。在这种情况下,模型反演方程简化为r中的一个立方方程,然后可以使用cardano的立方解公式来表示反演模型。
在所有其他情况下,使用各种算法求解非线性方程组,以数值方式反转模型。opencv使用迭代的“假位置”方法。
因为您想使用逆模型来消除一组图像的失真(这是正常的用例),所以应该使用initundortorrectitymap一次性地计算图像的不失真解决方案,然后将其传递给每个图像以重新Map以实际消除图像的失真。
如果你真的需要逆模型的参数模型,我的建议是用一对高阶多项式或薄板样条函数来近似initundortrectitymap返回的Map。