OpenCV: Как оценить расстояние по решению pnP?

Я попытался оценить расстояние между моей камерой и шахматной доской, которую я использовал для калибровки. Но у меня сильно различаются значения Z (993 - 1021 мм) для точек на шахматной доске. Шахматная доска параллельна камере, поэтому я ожидал, что значения Z между центром камеры и любой точкой на плоскости шахматной доски должны быть одинаковыми.

Что я сделал не так?

Код:

import numpy
import cv2

#from cv2.calibrateCamera.
imagePoints = numpy.array([[185.38619995, 88.96965027],
                           [215.52053833, 89.59921265],
                           [245.42134094, 90.34884644],
                           [275.23474121, 90.67272186],
                           [304.63644409, 91.27327728],
                           [333.90859985, 91.75786591],
                           [185.34030151, 118.99983978],
                           [215.52850342, 119.46482849],
                           [245.514328, 119.89020538],
                           [275.03085327, 120.42667389],
                           [304.64593506, 120.67990112],
                           [333.73968506, 121.14579773],
                           [185.34428406, 148.93850708],
                           [215.58241272, 149.32409668],
                           [245.4022522, 149.86160278],
                           [275.17852783, 149.81486511],
                           [304.5428772, 150.28009033],
                           [333.79421997, 150.35099792],
                           [185.18565369, 178.83206177],
                           [215.445755, 179.09082031],
                           [245.40261841, 179.29736328],
                           [275.07699585, 179.61761475],
                           [304.49923706, 179.56680298],
                           [333.78756714, 179.81796265],
                           [185.00271606, 208.8531189],
                           [215.40895081, 208.91560364],
                           [245.23666382, 209.10614014],
                           [275.06970215, 209.07598877],
                           [304.59561157, 209.13166809],
                           [333.90438843, 209.09313965],
                           [184.86560059, 238.93251038],
                           [215.23825073, 238.76199341],
                           [245.25619507, 238.73907471],
                           [274.97564697, 238.73266602],
                           [304.59933472, 238.72055054],
                           [333.96078491, 238.68598938],
                           [184.71801758, 269.20956421],
                           [215.05563354, 268.91809082],
                           [245.1418457, 268.70123291],
                           [274.9392395, 268.59295654],
                           [304.65649414, 268.48577881],
                           [333.98513794, 268.29562378],
                           [184.43510437, 299.72283936],
                           [214.71133423, 299.52212524],
                           [244.97724915, 298.88729858],
                           [275.03652954, 298.61529541],
                           [304.74026489, 298.40454102],
                           [334.23959351, 298.21008301]])

objectPoints = numpy.array([[0., 0., 0.],
                           [0.083, 0., 0.],
                           [0.16599999, 0., 0.],
                           [0.24899998, 0., 0.],
                           [0.33199999, 0., 0.],
                           [0.41499999, 0., 0.],
                           [0., 0.083, 0.],
                           [0.083, 0.083, 0.],
                           [0.16599999, 0.083, 0.],
                           [0.24899998, 0.083, 0.],
                           [0.33199999, 0.083, 0.],
                           [0.41499999, 0.083, 0.],
                           [0., 0.16599999, 0.],
                           [0.083, 0.16599999, 0.],
                           [0.16599999, 0.16599999, 0.],
                           [0.24899998, 0.16599999, 0.],
                           [0.33199999, 0.16599999, 0.],
                           [0.41499999, 0.16599999, 0.],
                           [0., 0.24899998, 0.],
                           [0.083, 0.24899998, 0.],
                           [0.16599999, 0.24899998, 0.],
                           [0.24899998, 0.24899998, 0.],
                           [0.33199999, 0.24899998, 0.],
                           [0.41499999, 0.24899998, 0.],
                           [0., 0.33199999, 0.],
                           [0.083, 0.33199999, 0.],
                           [0.16599999, 0.33199999, 0.],
                           [0.24899998, 0.33199999, 0.],
                           [0.33199999, 0.33199999, 0.],
                           [0.41499999, 0.33199999, 0.],
                           [0., 0.41499999, 0.],
                           [0.083, 0.41499999, 0.],
                           [0.16599999, 0.41499999, 0.],
                           [0.24899998, 0.41499999, 0.],
                           [0.33199999, 0.41499999, 0.],
                           [0.41499999, 0.41499999, 0.],
                           [0., 0.49799997, 0.],
                           [0.083, 0.49799997, 0.],
                           [0.16599999, 0.49799997, 0.],
                           [0.24899998, 0.49799997, 0.],
                           [0.33199999, 0.49799997, 0.],
                           [0.41499999, 0.49799997, 0.],
                           [0., 0.58099997, 0.],
                           [0.083, 0.58099997, 0.],
                           [0.16599999, 0.58099997, 0.],
                           [0.24899998, 0.58099997, 0.],
                           [0.33199999, 0.58099997, 0.],
                           [0.41499999, 0.58099997, 0.]])

cameraMatrix = numpy.array([[364.21551784, 0., 253.82602981],
                            [0., 363.51882052, 206.4493302],
                            [0., 0., 1.]])

distortionCoefficient = numpy.array([[0.09360905, -0.37480762, 0.00147107, -0.00132464, 0.25238534]])

Z = numpy.zeros((48))
retval,R, T = cv2.solvePnP(objectPoints,imagePoints,cameraMatrix,distortionCoefficient) #from chessboard coordinate space to camera coordinate space
R,jacobian = cv2.Rodrigues(R) #from R-vector to R-matrix
for i in range(0,48):
    point = numpy.dot(objectPoints[i],R) + numpy.matrix.transpose(T)
    Z[i] = point[0,2] * 1000 #Z-value to mm

print(Z.max() - Z.min())
print(Z)

Изображение с шахматной доски

1 ответ

Решение

Ну, похоже, это моя ошибка. Вероятно, плата была не совсем параллельна камере, хотя я пытался это исправить. Если мы посмотрим на вычисленные координаты Z, то увидим, что координата Z линейно уменьшается от нижнего правого угла к верхнему левому углу.

print(numpy.round(Z.reshape(8,6)))

[[ 1013.  1009.  1005.  1001.   997.   993.]
 [ 1014.  1010.  1006.  1002.   998.   994.]
 [ 1015.  1011.  1007.  1003.  1000.   996.]
 [ 1016.  1012.  1009.  1005.  1001.   997.]
 [ 1018.  1014.  1010.  1006.  1002.   998.]
 [ 1019.  1015.  1011.  1007.  1003.   999.]
 [ 1020.  1016.  1012.  1008.  1004.  1001.]
 [ 1021.  1017.  1013.  1010.  1006.  1002.]]
Другие вопросы по тегам