OpenCV 中的图像处理 006_图像渐变

本文主要内容来自于 OpenCV-Python 教程OpenCV 中的图像处理 部分,这部分的全部主要内容如下:

目标

在本章中,我们将学习:

理论

OpenCV 提供了三种类型的渐变滤波器或高通滤波器,Sobel、Scharr 和 Laplacian。我们将看下它们中的每一个。

1. Sobel 和 Scharr 导数

Sobel 算子是联合高斯平滑加微分运算,所以抗噪声能力更强。我们可以指定要采用的导数方向,垂直或水平(分别通过参数 yorderxorder)。你也可以通过参数 ksize 指定内核的大小。如果 ksize = -1,则将使用一个 3x3 Scharr 滤波器,相对于 3x3 Sobel 滤波器它能给出更好的结果。请参考所使用的内核的文档。

2. Laplacian 导数

它计算由关系 Δ s r c = ∂ 2 s r c ∂ x 2 + ∂ 2 s r c ∂ y 2 \Delta src = \frac{\partial ^2{src}}{\partial x^2} + \frac{\partial ^2{src}}{\partial y^2} Δsrc=x22src+y22src 给出的图像的拉普拉斯算子,其中每个导数都是使用 Sobel 导数找到的。如果 ksize = 1,则将使用如下的内核用于滤波
k e r n e l = [ 0 1 0 1 − 4 1 0 1 0 ] kernel = \begin{bmatrix} 0 & 1 & 0 \\ 1 & -4 & 1 \\ 0 & 1 & 0 \end{bmatrix} kernel=010141010

代码

下面的代码在一幅图中展示了所有这些操作。所有的内核都是 5x5 大小的。输出图像的深度传入 -1,以获得 np.uint8 类型的结果。

import numpy as np
import cv2 as cv
from matplotlib import pyplot as plt

def image_gradients():
    cv.samples.addSamplesDataSearchPath("/media/data/my_multimedia/opencv-4.x/samples/data")
    img = cv.imread(cv.samples.findFile('sudoku.png'), 0)
    print(img.shape)
    img = cv.resize(img, (359, 361))
    laplacian = cv.Laplacian(img, cv.CV_64F)
    sobelx = cv.Sobel(img, cv.CV_64F, 1, 0, ksize=5)
    sobely = cv.Sobel(img, cv.CV_64F, 0, 1, ksize=5)

    cv.imshow("Original", img)
    cv.imshow("Laplacian", laplacian)
    cv.imshow("Sobel X", sobelx)
    cv.imshow("Sobel Y", sobely)

    cv.waitKey(-1)
    cv.destroyAllWindows()


if __name__ == "__main__":
    image_gradients()

结果如下:
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-qZrKoTJl-1651362541111)(https://upload-images.jianshu.io/upload_images/1315506-5e604eab748f2c88.png)]

一个重要的问题

在我们的上一个例子中,输出数据类型是 cv.CV_8Unp.uint8。但这有一个小问题。黑色到白色的过渡被视为正斜率(它具有正值),而白色到黑色的过渡被视为负斜率(它具有负值)。因此当你将数据转为 np.uint8 时,所有的负斜率都被设置为了 0。简单来说,就是丢失了边缘。

如果我们想要探测两个边缘,更好的选项是保持输出数据类型为一些更高的形式,比如 cv.CV_16Scv.CV_64F 等等,取它的绝对值,然后将它转回 cv.CV_8U。下面的代码演示了水平 Sobel 滤波器的此过程和结果差异。

import numpy as np
import cv2 as cv
from matplotlib import pyplot as plt

def image_gradients2():
    img = np.zeros((600, 600))
    cv.rectangle(img, (150, 100), (450, 500), (255), -1)

    # Output dtype = cv.CV_8U
    sobelx8u = cv.Sobel(img, cv.CV_8U, 1, 0, ksize=5)

    # Output dtype = cv.CV_64F. Then take its absolute and convert to cv.CV_8U
    sobelx64f = cv.Sobel(img, cv.CV_64F, 1, 0, ksize=5)
    abs_sobel64f = np.absolute(sobelx64f)
    sobel_8u = np.uint8(abs_sobel64f)

    plt.subplot(1, 3, 1), plt.imshow(img, cmap='gray')
    plt.title('Original'), plt.xticks([]), plt.yticks([])
    plt.subplot(1, 3, 2), plt.imshow(sobelx8u, cmap='gray')
    plt.title('Sobel CV_8U'), plt.xticks([]), plt.yticks([])
    plt.subplot(1, 3, 3), plt.imshow(sobel_8u, cmap='gray')
    plt.title('Sobel abs(CV_64F)'), plt.xticks([]), plt.yticks([])

    plt.show()


if __name__ == "__main__":
    image_gradients2()

检查结果如下:
Image

其它资源

练习

参考文档

Image Gradients

Done.

好的,以下是基于OpenCV库的Python代码示例: ```python import cv2 import numpy as np # 读取两张图像 img1 = cv2.imread('image1.jpg') img2 = cv2.imread('image2.jpg') # 将两张图像转换为灰度图像 gray1 = cv2.cvtColor(img1, cv2.COLOR_BGR2GRAY) gray2 = cv2.cvtColor(img2, cv2.COLOR_BGR2GRAY) # 创建高斯金字塔 G1 = gray1.copy() G2 = gray2.copy() gp1 = [G1] gp2 = [G2] for i in range(6): G1 = cv2.pyrDown(G1) G2 = cv2.pyrDown(G2) gp1.append(G1) gp2.append(G2) # 创建拉普拉斯金字塔 lp1 = [gp1[5]] lp2 = [gp2[5]] for i in range(5, 0, -1): G1 = cv2.pyrUp(gp1[i]) G2 = cv2.pyrUp(gp2[i]) L1 = cv2.subtract(gp1[i - 1], G1) L2 = cv2.subtract(gp2[i - 1], G2) lp1.append(L1) lp2.append(L2) # 创建掩模 mask1 = np.zeros(gray1.shape, dtype=gray1.dtype) mask1[:, :int(gray1.shape[1] / 2)] = 255 mask2 = cv2.bitwise_not(mask1) # 对两个图像的金字塔进行融合 LS = [] for l1, l2 in zip(lp1, lp2): rows, cols = l1.shape ls = cv2.add(cv2.multiply(l1, mask1), cv2.multiply(l2, mask2)) LS.append(ls) # 重建融合图像 ls_ = LS[0] for i in range(1, 6): ls_ = cv2.pyrUp(ls_) ls_ = cv2.add(ls_, LS[i]) # 最终结果 result = cv2.cvtColor(ls_, cv2.COLOR_GRAY2BGR) cv2.imshow('result', result) cv2.waitKey(0) cv2.destroyAllWindows() ``` 这个代码示例,我们首先读取两张要融合的图像,并将它们转换为灰度图像。然后,我们构建了这两个图像的高斯金字塔和拉普拉斯金字塔。接着,我们创建了一个掩模,用于指定两张图像在融合过程的权重。我们对这两个图像的金字塔进行融合,并重建出最终的融合图像。最后,我们将结果转换为彩色图像,并显示出来。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值