目录
0 程序环境与所学函数
1 Prewitt、Robert 算子
?2 sobel 、scharr算子
3 Laplacian 、 LoG算子
?4 Canny边缘检测
5 Canny边缘检测底层代码实现
0 程序环境与所学函数
本章程序运行需要导入下面三个库,并定义了一个显示图像的函数
import cv2 as cv
import numpy as np
import matplotlib.pyplot as plt
?
def show(img):
? ? if img.ndim == 2:
? ? ? ? plt.imshow(img, cmap='gray')
? ? else:
? ? ? ? plt.imshow(cv.cvtColor(img, cv.COLOR_BGR2RGB))
? ? plt.show()
所学函数
Premitt、Roberts算子
cv.filter2D(img, -1, kernel)
Sobel、Scharr算子
cv.Sobel(img, cv.CV_16S, 1, 0, ksize=3)
cv.Scharr(img, cv.CV_64F, 0, 1, ksize=3)
Laplacian、LoG算子
cv.Laplacian(img, cv.CV_16S)
Canny边缘检测
cv.Canny(img, 20, 200)
1 Prewitt、Robert 算子
Prewitt原理:红色矩阵代表两个不同方向的算子(水平方向与竖直方向)
data:image/s3,"s3://crabby-images/7fcd5/7fcd5c145db67c6e1d48fc6e44c9bf397c4095d4" alt=""
效果
data:image/s3,"s3://crabby-images/53dcc/53dcc29d0202d1a203ec9654a3d5c0e04e667510" alt=""
?编程实现
# Prewitt算子
img = cv.imread('pic/gift500x500.jpg', 0)
kx = np.array([[1, 1, 1], [0, 0, 0], [-1, -1, -1]], dtype=np.float32)
ky = np.array([[-1,0, 1], [-1,0, 1], [-1, 0, 1]], dtype=np.float32)
imgx = cv.filter2D(img, cv.CV_16S, kx) #CV_16S是数据类型
imgy = cv.filter2D(img, cv.CV_16S, ky)
absX = cv.convertScaleAbs(imgx)
absY = cv.convertScaleAbs(imgy)
img_prewitt = cv.add(absX, absY) # np.uint8
show(img_prewitt)
显示
data:image/s3,"s3://crabby-images/8d1cc/8d1cc8830a9f42c11d86e5580e663292a584ac31" alt=""
# 锐化
img_pre_sharp1 = cv.addWeighted(img, 0.9, img_prewitt, 0.2, 0)
img_pre_sharp2 = cv.add(img, img_prewitt)
cv.imwrite('pic/gift_sharp.jpg', np.hstack([img, img_pre_sharp1, img_pre_sharp2]))
?Roberts算子原理
data:image/s3,"s3://crabby-images/2db64/2db6454317e791f55689716cfe5da84d65bb62b8" alt=""
?效果
data:image/s3,"s3://crabby-images/c382e/c382e46908f6359627f9ba3e59bdf7d2112968ca" alt=""
?程序实现
# Roberts算子
img = cv.imread('pic/gift500x500.jpg', 0)
kx = np.array([[-1, 0], [0, 1]], dtype=np.float32)
ky = np.array([[0, -1], [1, 0]], dtype=np.float32)
imgx = np.abs(cv.filter2D(img, cv.CV_16S, kx))
imgy = np.abs(cv.filter2D(img, cv.CV_16S, ky))
img_robert = cv.convertScaleAbs(np.abs(imgx) + np.abs(imgy)) # np.uint8
show(np.hstack([img, img_robert]))
显示
data:image/s3,"s3://crabby-images/66578/665785f73fb26df40d98f87411a1a2e9123d2b82" alt=""
?2 sobel 、scharr算子
原理
data:image/s3,"s3://crabby-images/23ee5/23ee56aa7698657f8385b4ae4722aaf738a084f7" alt="" data:image/s3,"s3://crabby-images/2db46/2db468c3df4977b582480e35c682037c4a8c3366" alt=""
?sobel算子编程实现
#sobel算子
img = cv.imread('pic/bookshelf500x333.jpg', 0)
show(img)
grad_x = cv.Sobel(img, cv.CV_16S, 1, 0, ksize=3)#1表示dx工作0表示dy不工作
grad_y = cv.Sobel(img, cv.CV_16S, 0, 1, ksize=3)
grad_xabs = np.abs(grad_x)
grad_yabs = np.abs(grad_y)
grad_xy = grad_xabs + grad_yabs
show(np.hstack([img, grad_xabs, grad_yabs, grad_xy]))
显示
data:image/s3,"s3://crabby-images/b9963/b99630e3d8fa8284c991bb34fbf6e77cfba1cb2c" alt=""
??Scharr算子编程实现
img = cv.imread('pic/bookshelf500x333.jpg', 0)
grad_x = np.abs(cv.Scharr(img, -1, 1, 0))
grad_y = np.abs(cv.Scharr(img, -1, 0, 1))
img_grad = cv.add(grad_x, abs(grad_y))
show(np.hstack([img, img_grad]))
show(np.hstack([grad_x, grad_y]))
显示
data:image/s3,"s3://crabby-images/34de1/34de1564d38b8bb3738eae6a72bb3b8131f21b01" alt=""
3 Laplacian 、 LoG算子
原理:
data:image/s3,"s3://crabby-images/dd81f/dd81f6e39bc3b46081f480211e46ada8c97b2467" alt=""
?data:image/s3,"s3://crabby-images/51d94/51d9405d1a3a88089eee6f4845bd840526916cbd" alt=""
?效果
data:image/s3,"s3://crabby-images/3fad9/3fad901b5660599df54b7fec9973a61d49d83803" alt=""
?Laplacian算子编程实现
img = cv.imread('pic/notebook500x333.jpg', 0)
img_lap = cv.Laplacian(img, cv.CV_16S, ksize=5, scale=0.2)
img_lap2 = np.abs(img_lap).clip(0, 255)
show(np.hstack([img, img_lap2]))
显示
data:image/s3,"s3://crabby-images/c2073/c20733173b2d88aef7f9922ee477f3ca89f30e5f" alt=""
?LoG算子编程实现
img = cv.imread('pic/notebook500x333.jpg', 0)
img_g = cv.GaussianBlur(img, (3,3), 0)
img_log = cv.Laplacian(img_g, cv.CV_16S, ksize=5, scale=0.2)
img_log2 = np.abs(img_log).clip(0, 255)
show(np.hstack([img, img_log2]))
显示
data:image/s3,"s3://crabby-images/39196/39196f40346d6a046694271764ed67a932508692" alt=""
?4 Canny边缘检测
原理:
data:image/s3,"s3://crabby-images/264f8/264f8c0544cc7de2d5b1b40aed436822904e895e" alt=""
?效果
data:image/s3,"s3://crabby-images/a6156/a61566c0784043133d85e3ec5a19762dea106433" alt=""
?编程实现
img = cv.imread('pic/notebook500x333.jpg', 0)
img_canny = cv.Canny(img, 20, 200) #小于20设为0大于200设为255
show(np.hstack([img, img_canny]))
显示
data:image/s3,"s3://crabby-images/430a5/430a580798651b3aebdd7742508b5a1d1ad579e8" alt=""
5 Canny边缘检测底层代码实现
原理
data:image/s3,"s3://crabby-images/a0201/a02010c1b73cf1e8e5ce30ae2d87ac6ea86e52c6" alt=""
?data:image/s3,"s3://crabby-images/c56cd/c56cdb22e301704cc8483890da544fea68a24c92" alt=""
?代码实现
img = cv.imread('pic/notebook500x333.jpg', 0)
# 1. 平滑
img_blur = cv.GaussianBlur(img, (5,5), 2)
# 2. 计算梯度
gradx = cv.Sobel(img_blur, cv.CV_64F, 1, 0)
grady = cv.Sobel(img_blur, cv.CV_64F, 0, 1)
R = np.abs(gradx) + np.abs(grady)
T = np.arctan(grady / (gradx + 1e-3))
# 3. 细化边缘
h, w = R.shape
img_thin = np.zeros_like(R)
for i in range(1, h-1):
for j in range(1, w-1):
theta = T[i,j]
if -np.pi / 8 <= theta < np.pi / 8:
if R[i,j] == max([R[i,j], R[i,j-1], R[i, j+1]]):
img_thin[i,j] = R[i,j]
elif -3 * np.pi / 8 <= theta < -np.pi / 8:
if R[i,j] == max([R[i,j], R[i-1,j+1], R[i+1,j-1]]):
img_thin[i,j] = R[i,j]
elif np.pi / 8 <= theta < 3 * np.pi / 8:
if R[i,j] == max([R[i,j], R[i-1,j-1], R[i+1,j+1]]):
img_thin[i,j] = R[i,j]
else:
if R[i,j] == max([R[i,j], R[i-1,j], R[i+1,j]]):
img_thin[i,j] = R[i,j]
show(img_thin)
th1 = 20
th2 = 200
h, w = img_thin.shape
img_edge = np.zeros_like(img_thin, dtype=np.uint8)
for i in range(1, h-1):
for j in range(1, w-1):
if img_thin[i,j] >= th2:
img_edge[i,j] = img_thin[i,j]
elif img_thin[i,j] > th1:
around = img_thin[i-1:i+2, j-1:j+2]
if around.max() >= th2:
img_edge[i,j] = img_thin[i,j]
show(img_edge)
?
?
|