OpenCV-Python小应用(八):判断是否有深色线条

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

前言

前提条件

相关介绍

  • Python是一种跨平台的计算机程序设计语言。是一个高层次的结合了解释性、编译性、互动性和面向对象的脚本语言。最初被设计用于编写自动化脚本(shell),随着版本的不断更新和语言新功能的添加,越多被用于独立的、大型项目的开发。
  • OpenCV是一个基于Apache2.0许可(开源)发行的跨平台计算机视觉和机器学习软件库,可以运行在Linux、Windows、Android和Mac OS操作系统上。它轻量级而且高效——由一系列C函数和少量C++类构成,同时提供了Python、Ruby、MATLAB等语言的接口,实现了图像处理和计算机视觉方面的很多通用算法。
  • OpenCV用C++语言编写,它具有C++、Python、Java和MATLAB接口,并支持Windows、Linux、Android和Mac OS,OpenCV主要倾向于实时视觉应用,并在可用时利用MMX和SSE指令。
  • 图像梯度是指图像中灰度变化的速度,反映了图像的边缘信息。在图像处理中,我们可以把图像看成二维离散函数,图像梯度其实就是这个二维离散函数的求导。
  • 在一幅模糊图像中的物体的轮廓不明显,轮廓边缘灰度变化不强烈,从而导致层次感不强,而在清晰图片中的物体轮廓边缘灰度变化明显,层次感强。因此,我们引入的图像梯度可以把图像看成二维离散函数,图像梯度其实就是这个二维离散函数的求导。
  • 在上边这幅图中可以看出,如果一副图像的相邻灰度值有变化,那么梯度就存在,如果图像相邻的像素没有变化,那么梯度就是0,把梯度值和相应的像素相加,那么灰度值没有变化的,像素就没有变化,灰度值变了,像素值也就变了。
  • 一些经典的图像梯度算法是考虑图像的每个像素的某个邻域内的灰度变化,利用边缘临近的一阶或二阶导数变化规律,对原始图像中像素某个邻域设置梯度算子,通常我们用小区域模板进行卷积来计算,有Sobel算子、Robinson算子、Laplace算子等。
  • 图像的灰度值是指图像中每个像素的亮度值,通常用于黑白图像。灰度值的范围通常是0到255,其中0表示黑色,255表示白色。在计算机视觉中,灰度图像是由纯黑和纯白来过渡得到的,在黑色中加入白色就得到灰色,纯黑和纯白按不同的比例来混合就得到不同的灰度值。
  • 在灰度图像中,每个像素的颜色值都是灰度值,指黑白图像中点的颜色深度,范围一般从0到255,白色为255,黑色为0。在灰度图像中,每个像素只有一个采样颜色的图像,这类图像通常显示为从最暗黑色到最亮的白色的灰度,尽管理论上这个采样可以任何颜色的不同深浅,甚至可以是不同亮度上的不同颜色。

实验环境

  • Python 3.6.13 (面向对象的高级语言)
  • OpenCV 3.4.10(python第三方库)pip3 install opencv-python==3.4.10.37

判断是否有深色线条

在这里插入图片描述

思路一:通过图像梯度直方图判断

  • 基本思路:对图像进行梯度计算,得到的梯度图像,进行每一行梯度值的统计,得到梯度直方图。如果梯度直方图有一定数量的高峰(这里阈值条件为高峰数量len(lst) > 3),则认为有深色线条,否则,认为无深色线条。
    在这里插入图片描述
  • 图像梯度相关知识点,可查阅OpenCV-Python快速入门(七):边缘检测
import  cv2
import matplotlib.pyplot as plt
import numpy as np
def get_magnitude(img):
    '''
    param: 
        img 图像数组
    return:
        magnitude 图像梯度
    '''
    # 高斯模糊,平滑
    Blur = cv2.GaussianBlur(img, (3, 3), 0, sigmaY=0, borderType=cv2.BORDER_REPLICATE) 
    # 计算图像x方向梯度
    sobel_dx = cv2.Sobel(Blur, cv2.CV_32F, 1, 0, ksize=3, scale=1, delta=0, borderType=cv2.BORDER_REPLICATE)
    # 计算图像y方向梯度
    sobel_dy = cv2.Sobel(Blur, cv2.CV_32F, 0, 1, ksize=3, scale=1, delta=0, borderType=cv2.BORDER_REPLICATE)
    # 得到图像梯度
    magnitude = np.sqrt(sobel_dx * sobel_dx + sobel_dy * sobel_dy) # 梯度幅值
    # cv2.imshow("magnitude",magnitude)
    # cv2.waitKey()
    # cv2.destroyAllWindows()
    return magnitude

if __name__ == "__main__":
        img = cv2.imread('3.png',0)
        magnitude = get_magnitude(img)
        
        print(magnitude)

        X_sum=np.sum(magnitude,axis=1)
        lst = list(filter(lambda x : x > 10000,X_sum))
        print(lst)
        if len(lst) > 3:
            print(True)
        else:
            print(False)

        x=np.arange(0,len(X_sum)) # x轴坐标值# print('x.shape:'x.shape)
        plt.figure()
        plt.subplot(1, 2, 1)
        plt.imshow(magnitude,cmap='gray')
        plt.subplot(1, 2, 2)
        plt.plot(x,X_sum,c ='r') # 参数c为color简写,表示颜色,r为red即红色# plt.show() # 显示图像
        plt.show()
True

在这里插入图片描述

思路二:通过图像灰度值变化判断

  • 基本思路:对图像的每一行求均值,对每一行均值以bin=5为一组,得到bin_mean,对每个相邻的bin_mean做差,如果差值大于一定阈值(这里阈值条件为max_dif > 3),则认为有深色线条,否则,认为无深色线条。

在这里插入图片描述

import cv2
import numpy as np
import matplotlib.pyplot as plt

img = cv2.imread('3.png',0)
X_mean = np.mean(img, axis=1)
print(X_mean.shape)
bin = 5
start = 0
lenght = len(X_mean)
bin_means = []
while start < lenght:
    end = start + bin
    if end > lenght:
        bin_mean = np.mean(X_mean[start:lenght])
    else:
        bin_mean = np.mean(X_mean[start:end])
    start += bin
    bin_means.append(bin_mean)

dif = abs(np.diff(np.array(bin_means))) # 相邻
max_dif = np.max(dif)
print(max_dif)
if max_dif > 3:
    print(True)
else:
    print(False)
True

参考

[1] https://opencv.org/
[2] 李立宗. OpenCV轻松入门:面向Python. 北京: 电子工业出版社,2019

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

FriendshipT

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值