OpenCV-下

模版匹配、霍夫变换

Posted by 新宇 on March 13, 2020

一、模版匹配与霍夫变换

1. 模版匹配

1.1 原理

所谓的模板匹配,就是在给定的图片中查找和模板最相似的区域,该算法的输入包括模板和图片,整个任务的思路就是按照滑窗的思路不断的移动模板图片,计算其与图像中对应区域的匹配度,最终将匹配度最高的区域选择为最终的结果。

1.2 流程

  • 将模板块每次移动一个像素 (从左往右,从上往下),在每一个位置,都计算与模板图像的相似程度。 对于每一个位置将计算的相似结果保存在结果矩阵(R)中。如果输入图像的大小(WxH)且模板图像的大小(wxh),则输出矩阵R的大小 为(W-w + 1,H-h + 1)将R显示为图像,如下图所示:

获得上述图像后,查找最大值所在的位置,那么该位置对应的区域就被认为是最匹配的。对应的区域就是以该点为顶点,长宽和模板图像一样大小的矩阵。

1.3 API

  • res = cv.matchTemplate(img,template,method)
    • img: 要进行模板匹配的图像
    • Template :模板
    • method:实现模板匹配的算法,主要有:
        1. 平方差匹配(CV_TM_SQDIFF):利用模板与图像之间的平方差进行匹配,最好的匹配是0,匹配越差,匹配的值越大。
        1. 相关匹配(CV_TM_CCORR):利用模板与图像间的乘法进行匹配,数值越大表示匹配程度较高,越小表示匹配效果差。
        1. 利用相关系数匹配(CV_TM_CCOEFF):利用模板与图像间的相关系数匹配,1表示完美的匹配,-1表示最差的匹配。

1.4 案例

import cv2 as cv
import numpy as np
from matplotlib import pyplot as plt
# 1 图像和模板读取

img = cv.imread('./image/wulin2.jpeg')
template = cv.imread('./image/wulin.jpeg')
h,w,l = template.shape
# 2 模板匹配
# 2.1 模板匹配

res = cv.matchTemplate(img, template, cv.TM_CCORR)
# 2.2 返回图像中最匹配的位置,确定左上角的坐标,并将匹配位置绘制在图像上 min_val, max_val, min_loc, max_loc = cv.minMaxLoc(res)
# 使用平方差时最小值为最佳匹配位置
# top_left = min_loc

top_left = max_loc
bottom_right = (top_left[0] + w, top_left[1] + h) 
cv.rectangle(img, top_left, bottom_right, (0,255,0), 2)
# 3 图像显示

plt.imshow(img[:,:,::-1])
plt.title('匹配结果'), plt.xticks([]), plt.yticks([])
plt.show()

1.5 模版匹配缺点

  • 模板匹配不适用于尺度变换,视角变换后的图像

2. 霍夫变换

2.1 原理

直角坐标系到极坐标系的转换

2.2 案例

import cv2 as cv
import matplotlib.pyplot as plt
# 1.加载图片,转为二值图

img = cv.imread('./image/rili.jpg')
gray = cv.cvtColor(img, cv.COLOR_BGR2GRAY) 
edges = cv.Canny(gray, 50, 150)
# 2.霍夫直线变换

lines = cv.HoughLines(edges, 0.8, np.pi / 180, 150) 
# 3.将检测的线绘制在图像上(注意是极坐标噢)

for line in lines:
rho, theta = line[0] a = np.cos(theta)
b = np.sin(theta) x0 = a * rho
y0 = b * rho
x1 = int(x0 + 1000 * (-b))
y1 = int(y0 + 1000 * (a))
x2 = int(x0 - 1000 * (-b))
y2 = int(y0 - 1000 * (a))
cv.line(img, (x1, y1), (x2, y2), (0, 255, 0))
# 4. 图像显示

plt.figure(figsize=(10,8),dpi=100) 
plt.imshow(img[:,:,::-1]),plt.title('霍夫变换线检测') 
plt.xticks([]), plt.yticks([])
plt.show()

2.3 霍夫圆检测

  • 霍夫梯度法将霍夫圆检测分为两个阶段,第一阶段检测圆心,第二阶段利用圆心推导出圆半径。
    • 圆心检测的原理:
      • 圆心是圆周法线的交汇处,设置一个阈值,在某点的相交的直线的条数大于这个阈值就认为该交汇点为圆心。
    • 圆半径确定原理:
      • 圆心到圆周上的距离(半径)是相同的,确定一个阈值,只要相同距离的数量大于该阈值,就认为该距离是该圆心的半径。
import cv2 as cv
import numpy as np
import matplotlib.pyplot as plt
# 1 读取图像,并转换为灰度图
planets = cv.imread("./image/star.jpeg")
gay_img = cv.cvtColor(planets, cv.COLOR_BGRA2GRAY)
# 2 进行中值模糊,去噪点
img = cv.medianBlur(gay_img, 7)
# 3 霍夫圆检测
circles = cv.HoughCircles(img, cv.HOUGH_GRADIENT, 1, 200, param1=100, param2=30, minRadius=0, maxRadius=100)
# 4 将检测结果绘制在图像上
for i in circles[0, :]: # 遍历矩阵每一行的数据
	# 绘制圆形
	cv.circle(planets, (i[0], i[1]), i[2], (0, 255, 0), 2) 
	# 绘制圆心
	cv.circle(planets, (i[0], i[1]), 2, (0, 0, 255), 3) 
# 5 图像显示
plt.figure(figsize=(10,8),dpi=100) 
plt.imshow(planets[:,:,::-1])
plt.title('霍夫变换圆检测') 
plt.xticks([]), plt.yticks([])
plt.show()