您好,登錄后才能下訂單哦!
魚眼鏡頭怎么利用OpenCV進行校準?相信很多沒有經(jīng)驗的人對此束手無策,為此本文總結(jié)了問題出現(xiàn)的原因和解決方法,通過這篇文章希望你能解決這個問題。
01.簡介
當(dāng)我們使用的魚眼鏡頭視角大于160°時,OpenCV中用于校準鏡頭“經(jīng)典”方法的效果可能就不是和理想了。
02.相機參數(shù)獲取
校準鏡頭其實只需要下面2個步驟。
利用OpenCV計算鏡頭的2個固有參數(shù)。OpenCV稱它們?yōu)镵和D,我們只需要知道它們是numpy數(shù)組外即可。
通過K和D對圖像進行去畸變矯正。
計算K和D
下載棋盤格圖案并將其打印在紙上(字母或A4尺寸)。大家要盡量將這張紙粘在堅硬且平坦的物體表面,例如一塊硬紙板上。因為這里的關(guān)鍵是直線必須是直線。
將圖案放在相機前面拍攝一些圖像,圖案要取在不同的位置和角度。這里的關(guān)鍵是圖案需要以不同的方式出現(xiàn)失真(以便OpenCV盡可能多地了解鏡頭相關(guān)參數(shù))。
現(xiàn)在我們只需要將此Python腳本片段復(fù)制到calibrate.py先前保存這些圖像的文件夾中的文件中,就可以對其進行命名。
import cv2 assert cv2.__version__[0] == '3', 'The fisheye module requires opencv version >= 3.0.0' import numpy as np import os import glob CHECKERBOARD = (6,9) subpix_criteria = (cv2.TERM_CRITERIA_EPS+cv2.TERM_CRITERIA_MAX_ITER, 30, 0.1) calibration_flags = cv2.fisheye.CALIB_RECOMPUTE_EXTRINSIC+cv2.fisheye.CALIB_CHECK_COND+cv2.fisheye.CALIB_FIX_SKEW objp = np.zeros((1, CHECKERBOARD[0]*CHECKERBOARD[1], 3), np.float32) objp[0,:,:2] = np.mgrid[0:CHECKERBOARD[0], 0:CHECKERBOARD[1]].T.reshape(-1, 2) _img_shape = None objpoints = [] # 3d point in real world space imgpoints = [] # 2d points in image plane. images = glob.glob('*.jpg') for fname in images: img = cv2.imread(fname) if _img_shape == None: _img_shape = img.shape[:2] else: assert _img_shape == img.shape[:2], "All images must share the same size." gray = cv2.cvtColor(img,cv2.COLOR_BGR2GRAY) # Find the chess board corners ret, corners = cv2.findChessboardCorners(gray, CHECKERBOARD, cv2.CALIB_CB_ADAPTIVE_THRESH+cv2.CALIB_CB_FAST_CHECK+cv2.CALIB_CB_NORMALIZE_IMAGE) # If found, add object points, image points (after refining them) if ret == True: objpoints.append(objp) cv2.cornerSubPix(gray,corners,(3,3),(-1,-1),subpix_criteria) imgpoints.append(corners) N_OK = len(objpoints) K = np.zeros((3, 3)) D = np.zeros((4, 1)) rvecs = [np.zeros((1, 1, 3), dtype=np.float64) for i in range(N_OK)] tvecs = [np.zeros((1, 1, 3), dtype=np.float64) for i in range(N_OK)] rms, _, _, _, _ = \ cv2.fisheye.calibrate( objpoints, imgpoints, gray.shape[::-1], K, D, rvecs, tvecs, calibration_flags, (cv2.TERM_CRITERIA_EPS+cv2.TERM_CRITERIA_MAX_ITER, 30, 1e-6) ) print("Found " + str(N_OK) + " valid images for calibration") print("DIM=" + str(_img_shape[::-1])) print("K=np.array(" + str(K.tolist()) + ")") print("D=np.array(" + str(D.tolist()) + ")")
運行python calibrate.py。如果一切順利,腳本將輸出如下內(nèi)容:
Found 36 images for calibration DIM=(1600, 1200) K=np.array([[781.3524863867165, 0.0, 794.7118000552183], [0.0, 779.5071163774452, 561.3314451453386], [0.0, 0.0, 1.0]]) D=np.array([[-0.042595202508066574], [0.031307765215775184], [-0.04104704724832258], [0.015343014605793324]])
03.圖像畸變矯正
獲得K和D后,我們可以對以下情況獲得的圖像進行失真矯正:我們需要取消失真的圖像與校準期間捕獲的圖像具有相同的尺寸。也可以將邊緣周圍的某些區(qū)域裁剪掉,來保證使未失真圖像的整潔。通過undistort.py使用以下python代碼創(chuàng)建文件:
DIM=XXX K=np.array(YYY) D=np.array(ZZZ) def undistort(img_path): img = cv2.imread(img_path) h,w = img.shape[:2] map1, map2 = cv2.fisheye.initUndistortRectifyMap(K, D, np.eye(3), K, DIM, cv2.CV_16SC2) undistorted_img = cv2.remap(img, map1, map2, interpolation=cv2.INTER_LINEAR, borderMode=cv2.BORDER_CONSTANT) cv2.imshow("undistorted", undistorted_img) cv2.waitKey(0) cv2.destroyAllWindows() if __name__ == '__main__': for p in sys.argv[1:]: undistort(p)
現(xiàn)在運行python undistort.py file_to_undistort.jpg。
看完上述內(nèi)容,你們掌握魚眼鏡頭怎么利用OpenCV進行校準的方法了嗎?如果還想學(xué)到更多技能或想了解更多相關(guān)內(nèi)容,歡迎關(guān)注億速云行業(yè)資訊頻道,感謝各位的閱讀!
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。