現存在一個問題,就下面圖片中的兩本書而言,怎樣快速讓中間邊的書本與左邊書本對齊(最終效果能實現兩張圖片重疊(最終結果爲右圖)),進行的圖像轉變可旋轉、平移、縮放、形變。web
本文主要內容就是介紹利用 Opencv 來怎樣解決上面的問題,解決這個問題須要三步微信
肯定至少四組對應點座標app
找到一個轉換矩陣;函數
把找到的轉換矩陣應用到 Moving Image(須要移動的圖片) 上,實現圖像對齊spa
圖片旋轉、平移、縮放等操做的主要目的,就是要最終實現兩圖像中點對點一一映射關係,圖像映射本質上就是像素點轉換.net
圖中標記了其中四組對應點,分別標爲不一樣的顏色,分別標爲紅、橙、黃和綠四種顏色;好比這裏的 和 是就是一組對應點,圖片通過轉換以後 點 必須映射到 點位置。3d
涉及圖片中點座標變換,都須要藉助於 矩陣 運算,這裏探究的圖像維度都屬於二維,座標只須要 便可code
面向此類轉換問題,Homography 轉換 ( 3 × 3 矩陣) 可用於解決此類轉化問題,用來解決點對點映射問題,Homography 矩陣可寫做下列方式:
則 、 做爲對應點,則 Homography 的的應用 以下:
而矩陣 H 參數的肯定至少須要 4 組對應點,所以在計算 H 時至少要找到 4 組對應點;找到的對應點組數越多,計算獲得的 H 會越精確,最終的轉換效果也就會越好。orm
下面用 Opencv + Python 來實現上面圖片中的書籍的對齊,blog
import cv2
import numpy as np
if __name__ =='__main__':
#圖片讀取
img_src = cv2.imread("D:/book2.jpg")
position_src = np.array([[141,131],[480,159],[493,630],[64,601]],dtype = float)
img_dst = cv2.imread("D:/book1.jpg")
position_dst = np.array([[318,256],[543,372],[316,670],[73,473]],dtype = float)
#計算轉換矩陣
h,status = cv2.findHomography(position_src,position_dst)
#對圖片進行仿射變換
out_img = cv2.warpPerspective(img_src,h,(img_dst.shape[1],img_dst.shape[0]))
#Display images;
cv2.imshow("Source image",img_src)
cv2.imshow("Destination Image",img_dst)
cv2.imshow("Warped Source Image",out_img)
cv2.waitKey(0)
這裏事先已經肯定好對應的四個點的座標,而後把這四個點的座標帶入 cv2.findHomography() 計算出轉換矩陣,最後把矩陣應用到兩圖像中,獲得最終的轉換結果,
這裏提醒一點,warpPerspective 函數進行對圖像像素進行矩陣變換時,隱藏了一個參數 Interpolator ,默認爲線性插值,功能是防止像素點像素值缺失
上面小案例不方便的一點須要肯定對應四個點的座標,這個步驟是比較繁瑣的,下面案例將在程序中加入交互功能,實現某個圖片的自動標記點收集、標記點點轉換:
首先須要準備兩張圖片,其中一張爲海報,一張爲須要替換的海報;關於肯定點的座標時,被替換的圖片的座標很是好肯定,只需知道圖片的長寬便可;
但的海報圖像區域四個點是很差肯定的, 這裏利用 Opencv 的鼠標回調函數,監視鼠標響應,根據用戶點擊來收集 PIck 獲得的座標;
def mouse_handler(event,x,y,flags,data):
if event ==cv2.EVENT_LBUTTONDOWN:
cv2.circle(data['im'],(x,y),3,(0,0,255),5,16)
cv2.namedWindow("Image",0)
cv2.imshow("Image",data['im'])
if len(data['points']) <4:
data['points'].append([x,y])
def get_four_points(im):
data = {}
data['im'] = im.copy()
data['points'] = []
# Set the callback function for any mouse event
cv2.namedWindow("Image", 0)
cv2.imshow('Image',im)
#請注意你標記點的數據,是順時針,須要與pst_src 方向一致
cv2.setMouseCallback("Image",mouse_handler,data)
cv2.waitKey(0)
# Convert array to np.array
#豎直方向堆疊起來;;;
points = np.vstack(data['points']).astype(float)
return points
座標肯定之後,接下來就很簡單了,跟上個案例同樣,計算變換矩陣,矩陣應用到圖像旋轉,最終更換海報內容也就輕鬆完成啦
須要注意一點,座標 Pick 點的順序須與記錄替換圖像頂點順序一致,不然轉換圖會有誤差,案例完整代碼以下:
if __name__ =='__main__':
img_src = cv2.imread("D:/first-image.jpg")
size = img_src.shape
# 取得四個座標
pst_src = np.array(
[
[0,0],[size[1]-1,0],
[size[1]-1,size[0]-1],
[0,size[0]-1]
],dtype=float
)
#Read the destination image
img_dst = cv2.imread("D:/times-square.jpg")
print("Click on four corners of bllboard and the press ENTER")
four_point = get_four_points(img_dst)
# Calculate Homography between source and destination points
h,status = cv2.findHomography(pst_src,four_point)
im_temp = cv2.warpPerspective(img_src,h,(img_dst.shape[1],img_dst.shape[0]))
cv2.fillConvexPoly(img_dst,four_point.astype(int),0,16)
#add wraped source image to destination image
img_dst = img_dst + im_temp
cv2.namedWindow("Image", 0)
cv2.imshow("Image",img_dst)
cv2.waitKey(0)
推薦閱讀:
本文分享自微信公衆號 - Z先生點記(gh_683d048a482a)。
若有侵權,請聯繫 support@oschina.cn 刪除。
本文參與「OSC源創計劃」,歡迎正在閱讀的你也加入,一塊兒分享。