5.opencv——图像变换3(阈值处理-二值化,反二值化,截断阈值处理,超阈值零,低阈值零处理,Otsu算法,三角算法)

本文主要是介绍5.opencv——图像变换3(阈值处理-二值化,反二值化,截断阈值处理,超阈值零,低阈值零处理,Otsu算法,三角算法),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

阈值处理

  • 阈值处理
    • 全局阈值处理
      • 二值化阈值处理
      • 反二值阈值处理
      • 截断阈值处理
      • 超阈值零处理
      • 低阈值零处理
      • Otsu算法阈值处理
      • 三角算法阈值处理
    • 自适应阈值处理

阈值处理

\qquad 阈值化处理:以某种规则依次将像素处理成0或1输出,即像素分割,可以被视作最简单的图像分割方法。给定一个灰度图,经过阈值化处理将把它转化成二值图输出。阈值化处理提取背景中的重要信息,将图像更进一步的从灰度图背景中抽离出来。利用阈值进行分割的方法是基于图像中物体和背景之间的灰度差异来实现的。阈值处理用于剔除图像中像素值高与或者低于指定值的像素点。

全局阈值处理

\qquad 全局阈值处理是指将大于阈值的像素值设置为 255,将其他像素值设置为 0;或者将大于阈值的像素值设置为 0,将其他像素值设置为 255。
OpenCV 的 cv2.threshold()函数用于实现全局阈值处理,其基本格式如下:

retval, dst=cv2.threshold(src, thresh, maxval, type)

其参数说明如下:

参数说明
retval为返回的阈值
dst为全局阈值处理后的结果图像
src为原图像
thresh为设置的阈值
maxval是阈值类型为THRESH_BINARY和THRESH_BINARY _INV 时使用的最大值
type为阈值类型

二值化阈值处理

\qquad cv2.threshold()函数的 type 参数值为cv2.THRESH_BINARY 时执行二值化阈值处理,将大于阈值的像素值设置为 255,将其他像素值设置为 0。
以一个大小为 3 × 3 3\times3 3×3的图像像素为例子,将阈值设置为150。
原图:
在这里插入图片描述
二值化阈值处理后结果:
在这里插入图片描述

import cv2img = cv2.imread('bee.jpg')
cv2.imshow('img', img)
ret, img2 = cv2.threshold(img, 150, 255, cv2.THRESH_BINARY)  # 阈值处理
cv2.imshow('imgTHRESH_BINARY', img2)
cv2.waitKey(0)

在这里插入图片描述

反二值阈值处理

\qquad cv2.threshold()函数的 type 参数值为cv2.THRESH_BINARY_INV 时执行反二值化阈值处理,将大于阈值的像素值设置为 0,将其他像素值设置为 255。
以一个大小为 3 × 3 3\times3 3×3的图像像素为例子,将阈值设置为150。
原图:
在这里插入图片描述
反二值化阈值处理后结果:
在这里插入图片描述

import cv2img = cv2.imread('bee.jpg')
cv2.imshow('img', img)
ret, img2 = cv2.threshold(img, 150, 255, cv2.THRESH_BINARY_INV)  # 阈值处理
cv2.imshow('imgTHRESH_BINARY_INV', img2)
cv2.waitKey(0)

在这里插入图片描述

截断阈值处理

\qquad cv2.threshold()函数的 type 参数值为cv2.THRESH_TRUNC时执行截断阈值处理,将大于阈值的像素值设置为 阈值,将其他像素值保持不变。
以一个大小为 3 × 3 3\times3 3×3的图像像素为例子,将阈值设置为150。
原图:
在这里插入图片描述
截断阈值处理后结果:
在这里插入图片描述

import cv2img = cv2.imread('bee.jpg')
cv2.imshow('img', img)
ret, img2 = cv2.threshold(img, 150, 255, cv2.THRESH_TRUNC)  # 阈值处理
cv2.imshow('imgTHRESH_TRUNC', img2)
cv2.waitKey(0)

在这里插入图片描述

超阈值零处理

\qquad cv2.threshold()函数的 type 参数值为cv2.THRESH_TOZERO时执行超阈值零处理,将大于阈值的像素值设置为0,将其他像素值保持不变。
以一个大小为 3 × 3 3\times3 3×3的图像像素为例子,将阈值设置为150。
原图:
在这里插入图片描述
超阈值零处理后结果:
在这里插入图片描述

import cv2img = cv2.imread('bee.jpg')
cv2.imshow('img', img)
ret, img2 = cv2.threshold(img, 150, 255, cv2.THRESH_TOZERO)  # 阈值处理
cv2.imshow('imgTHRESH_TOZERO', img2)
cv2.waitKey(0)

在这里插入图片描述

低阈值零处理

\qquad cv2.threshold()函数的 type 参数值为cv2.THRESH_TOZERO_INV时执行低阈值零处理,将小于阈值的像素值设置为0,将其他像素值保持不变。
以一个大小为 3 × 3 3\times3 3×3的图像像素为例子,将阈值设置为150。
原图:
在这里插入图片描述
低阈值零处理后结果:

在这里插入图片描述

import cv2img = cv2.imread('bee.jpg')
cv2.imshow('img', img)
ret, img2 = cv2.threshold(img, 150, 255, cv2.THRESH_TOZERO_INV)  # 阈值处理
cv2.imshow('imgTHRESH_TOZERO_INV', img2)
cv2.waitKey(0)

在这里插入图片描述

Otsu算法阈值处理

\qquad 对于色彩不均衡的图像,Otsu算法阈值处理的方式更好,它会遍历当前图像的所有阈值,在选择最佳阈值。
\qquad OTSU算法也称最大类间差法,有时也称之为大津算法,由大津于1979年提出,被认为是图像分割中阈值选取的最佳算法,计算简单,不受图像亮度和对比度的影响,因此在数字图像处理上得到了广泛的应用。它是按图像的灰度特性,将图像分成背景和前景两部分。因方差是灰度分布均匀性的一种度量,背景和前景之间的类间方差越大,说明构成图像的两部分的差别越大,当部分前景错分为背景或部分背景错分为前景都会导致两部分差别变小。因此,使类间方差最大的分割意味着错分概率最小。
原理:
\qquad 对于图像I(x,y),前景(即目标)和背景的分割阈值记作T,属于前景的像素点数占整幅图像的比例记为ω0,其平均灰度μ0;背景像素点数占整幅图像的比例为ω1,其平均灰度为μ1。图像的总平均灰度记为μ,类间方差记为g。
\qquad 假设图像的背景较暗,并且图像的大小为M×N,图像中像素的灰度值小于阈值T的像素个数记作N0,像素灰度大于阈值T的像素个数记作N1,则有:
       \qquad \qquad \qquad ω0=N0/ M×N (1)
       \qquad \qquad \qquad ω1=N1/ M×N (2)
       \qquad \qquad \qquad N0+N1=M×N (3)
       \qquad \qquad \qquad ω0+ω1=1    (4)
       \qquad \qquad \qquad μ = ω 0 ∗ μ 0 + ω 1 ∗ μ 1 ( 5 ) μ=ω0*μ0+ω1*μ1(5) μ=ω0μ0+ω1μ1(5)
     g = w 0 ( μ 0 − μ ) 2 + w 1 ( μ 1 − μ ) 2 ( 6 ) g = w_0(\mu_0-\mu)^2+w_1(\mu_1-\mu)^2 (6) g=w0(μ0μ)2+w1(μ1μ)2(6)
将式(5)代入式(6),得到等价公式:
       \qquad \qquad \qquad g=ω0ω1(μ0-μ1)^2    (7) 这就是类间方差
采用遍历的方法得到使类间方差g最大的阈值T,即为所求。

举个例子:以一个大小为 3 × 3 3\times3 3×3的图像像素为例子:
在这里插入图片描述
先以第一个像素为阈值T,即T=1,然后根据上面公式计算类间方差g
ω 0 = 1 9 ω 1 = 8 9 μ 0 = 1 μ 1 = 2 + 3 + 4 + 5 + 6 + 7 + 8 + 9 9 = 44 9 μ = 1 9 × 1 + 44 9 × 8 9 = 361 81 g = 1 9 × 8 9 × ( 1 − 44 9 ) 2 ≈ 1.5 \omega_0 =\frac {1} {9}\qquad\omega_1 =\frac {8} {9}\\ \mu_0 = 1\qquad\mu_1 = \frac {2+3+4+5+6+7+8+9} {9}=\frac {44} {9}\\ \mu = \frac {1} {9}\times1+\frac {44} {9}\times\frac {8} {9}=\frac {361} {81}\\ g = \frac {1} {9}\times\frac {8} {9}\times(1-\frac {44} {9})^2\approx1.5 ω0=91ω1=98μ0=1μ1=92+3+4+5+6+7+8+9=944μ=91×1+944×98=81361g=91×98×(1944)21.5
逐个计算得到每个像素点的g:

像素点g值
11.5
21.7
32
45
55
64.5
73.5
82
90

其中像素值为4和5 的类间方差(g)最大,所以最佳阈值为4或5。
总结

应用是求图像全局阈值的最佳方法,应用不言而喻,适用于大部分需要求图像全局阈值的场合
优点计算简单快速,不受图像亮度和对比度的影响
缺点对图像噪声敏感;只能针对单一目标分割;当目标和背景大小比例悬殊、类间方差函数可能呈现双峰或者多峰(上面的例子就出现了双峰情况),这个时候效果不好

cv2.threshold()函数的 type 参数值为cv2.THRESH_OTSU来实现Otsu算法阈值处理

import cv2img = cv2.imread('bee.jpg', cv2.IMREAD_GRAYSCALE)   # 读取图像,转化为单通道灰度图像
cv2.imshow('img', img)                              # 显示原图
ret, img2 = cv2.threshold(img, 127, 255, cv2.THRESH_BINARY)                      # 阈值处理
cv2.imshow('img2', img2)
ret, img3 = cv2.threshold(img, 127, 255, cv2.THRESH_BINARY+cv2.THRESH_OTSU)      # 阈值处理
cv2.imshow('img3', img3)
ret, img4 = cv2.threshold(img, 127, 255, cv2.THRESH_BINARY_INV+cv2.THRESH_OTSU)  # 阈值处理
cv2.imshow('img4', img4)
cv2.waitKey(0)

在这里插入图片描述

三角算法阈值处理

\qquad 三角法求阈值最早见于Zack的论文《Automatic measurement of sister chromatid exchange frequency》主要是用于染色体的研究,该方法是使用直方图数据,基于纯几何方法来寻找最佳阈值,它的成立条件是假设直方图最大波峰在靠近最亮的一侧,然后通过三角形求得最大直线距离,根据最大直线距离对应的直方图灰度等级即为分割阈值,图示如下:
在这里插入图片描述
对上图的详细解释:
在直方图上从最高峰处bmx到最暗对应直方图bmin(p=0)%构造一条直线,从bmin处开始计算每个对应的直方图b到直线的垂直距离,知道bmax为止,其中最大距离对应的直方图位置即为图像二值化对应的阈值T。

扩展情况:
有时候最大波峰对应位置不在直方图最亮一侧,而在暗的一侧,这样就需要翻转直方图,翻转之后求得值,用255减去即得到为阈值T。扩展情况的直方图表示如下:
请添加图片描述
上面引用https://www.codenong.com/cs105846920/
算法步骤

  1. 图像转灰度
  2. 计算图像灰度直方图
  3. 寻找直方图中两侧边界
  4. 寻找直方图最大值
  5. 检测是否最大波峰在亮的一侧,否则翻转
  6. 计算阈值得到阈值T,如果翻转则255-T

cv2.threshold()函数的 type 参数值为cv2.THRESH_TRIANGLE来实现三角算法阈值处理

import cv2img = cv2.imread('bee.jpg', cv2.IMREAD_GRAYSCALE)            # 读取图像,转化为单通道灰度图像
cv2.imshow('img', img)                            
ret, img2 = cv2.threshold(img, 127, 255, cv2.THRESH_BINARY)  # 阈值处理
cv2.imshow('img2', img2)
ret, img3 = cv2.threshold(img, 127, 255, cv2.THRESH_BINARY+cv2.THRESH_OTSU)
cv2.imshow('img3', img3)
ret, img4 = cv2.threshold(img, 127, 255, cv2.THRESH_BINARY_INV+cv2.THRESH_TRIANGLE)
cv2.imshow('img4', img4)
cv2.waitKey(0)

在这里插入图片描述

自适应阈值处理

\qquad 自适应阈值处理也称局部阈值处理,它通过计算每个像素点邻域的加权平均值来确定阈值,并用该阈值处理当前像素点。全局阈值处理适用于色彩均衡的图像,自适应阈值处理则适用于明暗差异较大的图像。
OpenCV 的 cv2.adaptiveThreshola()函数用于实现自适应阈值处理,其基本格式如下:

dst=cv2,adaptiveThreshold(src,maxValue,adaptiveMethod,thresholdType,blockSize,C)

其参数说明如下:

参数说明
dst为阈值处理的结果图像
src为原图像
maxValue为最大值
adaptiveMethod为自适应方法,其值为 CV2.ADAPTIVE_THRESH_MEAN_C(邻域中所有像素点的权重值相同)或者 cv2.ADAPTNVE_THRESHL_GAUSSIANL_C(邻域中像素点的权重值与其到中心点的距离有关,通过高斯方程可计算各个点的权重值)
thresholdType为阈值处理方式,其值为 cv2.THRESH_ BINARY(二值化阈值处理)或者 cv2.THRESH_BINARY_INV(反二值化阈值处理)
blockSize为计算局部阈值的邻域的大小。
C为常量,自适应阈值为 blockSize 指定邻域的加权平均值减去C
import cv2img = cv2.imread('bee.jpg', cv2.IMREAD_GRAYSCALE)  # 读取图像,转化为单通道灰度图像
cv2.imshow('img', img)
img2 = cv2.adaptiveThreshold(img, 255, cv2.ADAPTIVE_THRESH_MEAN_C,cv2.THRESH_BINARY, 5, 10)			# 阈值处理
cv2.imshow('img2', img2)
cv2.waitKey(0)

在这里插入图片描述

这篇关于5.opencv——图像变换3(阈值处理-二值化,反二值化,截断阈值处理,超阈值零,低阈值零处理,Otsu算法,三角算法)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/165134

相关文章

Python FastAPI+Celery+RabbitMQ实现分布式图片水印处理系统

《PythonFastAPI+Celery+RabbitMQ实现分布式图片水印处理系统》这篇文章主要为大家详细介绍了PythonFastAPI如何结合Celery以及RabbitMQ实现简单的分布式... 实现思路FastAPI 服务器Celery 任务队列RabbitMQ 作为消息代理定时任务处理完整

C#使用SQLite进行大数据量高效处理的代码示例

《C#使用SQLite进行大数据量高效处理的代码示例》在软件开发中,高效处理大数据量是一个常见且具有挑战性的任务,SQLite因其零配置、嵌入式、跨平台的特性,成为许多开发者的首选数据库,本文将深入探... 目录前言准备工作数据实体核心技术批量插入:从乌龟到猎豹的蜕变分页查询:加载百万数据异步处理:拒绝界面

opencv图像处理之指纹验证的实现

《opencv图像处理之指纹验证的实现》本文主要介绍了opencv图像处理之指纹验证的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学... 目录一、简介二、具体案例实现1. 图像显示函数2. 指纹验证函数3. 主函数4、运行结果三、总结一、

Springboot处理跨域的实现方式(附Demo)

《Springboot处理跨域的实现方式(附Demo)》:本文主要介绍Springboot处理跨域的实现方式(附Demo),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不... 目录Springboot处理跨域的方式1. 基本知识2. @CrossOrigin3. 全局跨域设置4.

SpringBoot实现MD5加盐算法的示例代码

《SpringBoot实现MD5加盐算法的示例代码》加盐算法是一种用于增强密码安全性的技术,本文主要介绍了SpringBoot实现MD5加盐算法的示例代码,文中通过示例代码介绍的非常详细,对大家的学习... 目录一、什么是加盐算法二、如何实现加盐算法2.1 加盐算法代码实现2.2 注册页面中进行密码加盐2.

python+opencv处理颜色之将目标颜色转换实例代码

《python+opencv处理颜色之将目标颜色转换实例代码》OpenCV是一个的跨平台计算机视觉库,可以运行在Linux、Windows和MacOS操作系统上,:本文主要介绍python+ope... 目录下面是代码+ 效果 + 解释转HSV: 关于颜色总是要转HSV的掩膜再标注总结 目标:将红色的部分滤

Python实现自动化接收与处理手机验证码

《Python实现自动化接收与处理手机验证码》在移动互联网时代,短信验证码已成为身份验证、账号注册等环节的重要安全手段,本文将介绍如何利用Python实现验证码的自动接收,识别与转发,需要的可以参考下... 目录引言一、准备工作1.1 硬件与软件需求1.2 环境配置二、核心功能实现2.1 短信监听与获取2.

Java时间轮调度算法的代码实现

《Java时间轮调度算法的代码实现》时间轮是一种高效的定时调度算法,主要用于管理延时任务或周期性任务,它通过一个环形数组(时间轮)和指针来实现,将大量定时任务分摊到固定的时间槽中,极大地降低了时间复杂... 目录1、简述2、时间轮的原理3. 时间轮的实现步骤3.1 定义时间槽3.2 定义时间轮3.3 使用时

Python使用date模块进行日期处理的终极指南

《Python使用date模块进行日期处理的终极指南》在处理与时间相关的数据时,Python的date模块是开发者最趁手的工具之一,本文将用通俗的语言,结合真实案例,带您掌握date模块的六大核心功能... 目录引言一、date模块的核心功能1.1 日期表示1.2 日期计算1.3 日期比较二、六大常用方法详

利用Go语言开发文件操作工具轻松处理所有文件

《利用Go语言开发文件操作工具轻松处理所有文件》在后端开发中,文件操作是一个非常常见但又容易出错的场景,本文小编要向大家介绍一个强大的Go语言文件操作工具库,它能帮你轻松处理各种文件操作场景... 目录为什么需要这个工具?核心功能详解1. 文件/目录存javascript在性检查2. 批量创建目录3. 文件