OpenCV-Python小应用(九):通过灰度直方图检测图像异常点

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到255,其中0表示黑色,255表示白色。在计算机视觉中,灰度图像是由纯黑和纯白来过渡得到的,在黑色中加入白色就得到灰色,纯黑和纯白按不同的比例来混合就得到不同的灰度值。
  • 在灰度图像中,每个像素的颜色值都是灰度值,指黑白图像中点的颜色深度,范围一般从0到255,白色为255,黑色为0。在灰度图像中,每个像素只有一个采样颜色的图像,这类图像通常显示为从最暗黑色到最亮的白色的灰度,尽管理论上这个采样可以任何颜色的不同深浅,甚至可以是不同亮度上的不同颜色。
  • 灰度直方图是一种用来描述数字图像中灰度级分布的图形工具。它是一个灰度级的函数,表示图像中具有某种灰度级的像素的个数,反映了图像中每种灰度出现的频率。
  • 灰度直方图是图像处理中非常重要的一个概念,它是我们对图像本身灰度的一个分析以及之后我们需要做二值化的一个基础的概念
  • 灰度直方图可以用来分析图像的对比度、亮度等特征,也可以用于图像增强、图像分割、图像压缩等领域。
  • 图像灰度直方图相关知识点,可查阅OpenCV-Python快速入门(九):直方图

实验环境

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

通过灰度直方图检测图像异常点

  • 基本思路:通过计算图像灰度,得到灰度直方图,这里认为灰度直方图中灰度值像素个数较少的灰度值大概率为异常点。
    在这里插入图片描述
  • 目录结构
    在这里插入图片描述

在这里插入图片描述

代码实现

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

def cal_low_high_hist(img):
    hist = cv2.calcHist([img],[0],None,[256],[0,255])
    flattened_hist = hist.flatten()  # [[255],[125],...] -> [255,125,...]
    # print(sum(flattened_hist)*0.125) # 下分位
    # print(sum(flattened_hist)*(1-0.125)) # 上分位
    # plt.plot(hist,color='b')
    # plt.show()
    
    # 下分位
    nums = 0
    for i in range(len(flattened_hist)):
        low_sum = sum(flattened_hist)*0.125
        nums += flattened_hist[i]
        if nums > low_sum:
            # print(i)
            break
            
    # 上分位
    nums = 0
    for j in range(len(flattened_hist)):
        low_sum = sum(flattened_hist)*(1-0.125)
        nums += flattened_hist[j]
        if nums > low_sum:
            # print(j)
            break
            
    return i,j

if __name__=="__main__":
    # 上分位计算值
    up_cal_val = [15,255]
    
    # 下分位计算值
    low_cal_val = [-255,-15]
    
    img_dir = 'imgs'
    img_name_list = os.listdir(img_dir)

    output_folder = 'out_imgs'
    if not os.path.exists(output_folder):
        os.mkdir(output_folder)

    for img_name in  img_name_list:
        img_path = os.path.join(img_dir,img_name)
        img = cv2.imread(img_path,0)
        i,j = cal_low_high_hist(img) # 下,上
        # print(i,j)
        up_cal_res = list(map(lambda x: x + j, up_cal_val))
        low_cal_res = list(map(lambda x: x + i, low_cal_val))
        # print(up_cal_res,low_cal_res) 
        up_cal_res = [x if x <= 255 else 255 for x in up_cal_res] # 亮点(异常点)
        low_cal_res = [x if x >= 0 else 0 for x in low_cal_res] # 暗点(异常点)
        # print(up_cal_res,low_cal_res) 
        
        #找到满足条件的像素点
        condition = np.logical_or(img>min(up_cal_res),img<max(low_cal_res))

        out_of_range_pixels = np.count_nonzero(condition)
        # print(out_of_range_pixels)
        # total_out_of_range_pixels += out_of_range_pixels
        
        small_image = copy.deepcopy(img)

        if out_of_range_pixels>4:
            contours,_ = cv2.findContours(condition.astype(np.uint8),cv2.RETR_EXTERNAL,cv2.CHAIN_APPROX_SIMPLE)

            valid_rectangles=[]
            for i,c in enumerate(contours):
                box=cv2.boundingRect(c)
                x,y,w,h, =box
                center_x=x+w//2
                center_y=y+h//2
                # valid_rectangles.append((x,y,w,h,center_x,center_y))
                valid_rectangles.append([x,y,w,h,center_x,center_y])
                # cv2.drawContours(small_image,contours,i,(0,255.0),2)
                # cv2.rectangle(small_image,(x,y),(x+w,y+h),(0,0,255),2)

            pad = 5
            for i,(x,y,w,h,_,_) in enumerate(valid_rectangles):
                cv2.rectangle(small_image,(x-pad,y-pad),(x+w+pad,y+h+pad),(0,0,255),1)
                

            # 保存图片
            output_filename = f"{os.path.splitext(img_name)[0]}_res.jpg"
            output_path = os.path.join(output_folder,output_filename)
            cv2.imwrite(output_path,small_image)

            # 显示图片
            show_img = cv2.resize(small_image,(small_image.shape[1],small_image.shape[0]))
            cv2.imshow('{}'.format(output_filename),show_img)
            cv2.waitKey()
            cv2.destroyAllWindows()    

输出结果

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

参考

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

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

FriendshipT

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

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

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

打赏作者

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

抵扣说明:

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

余额充值