热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

白话文讲计算机视觉-第四讲

大家好,我们又见面了,今天我们主讲的内容只有一个,那就是腐蚀膨胀,我也懒惰一回啦,别的就不讲啦,哈哈哈。首先我来说一下什么是腐蚀:腐蚀腐蚀就是是将灰度值小的像素点给扩展了,用来去

大家好,我们又见面了,今天我们主讲的内容只有一个,那就是腐蚀膨胀,我也懒惰一回啦,别的就不讲啦,哈哈哈。

首先我来说一下什么是腐蚀:

腐蚀

腐蚀就是是将灰度值小的像素点给扩展了,用来去除比较亮的(灰度值大的)噪点。假如我们现在有一张灰度图和一个卷积核:

 

我们要用左边的6*6的图和右边的卷积核做卷积。其中,图片中黑色的矩形区域表示亮度大的位置,白色的小矩形区域表示亮度小的位置。我们这次做的卷积和我们上节课讲的可就不同啦。这次我们要做的是腐蚀,所以我们首先在卷积核上面定一个中心店,比如我们定在了红色小球的位置,接下来,我们要做的是把这个小球的位置首先与图片的第一行第一列对齐:

如果我们的卷积核底下的像素点都是黑色,那么我们就把红色小球那一点的像素变为黑色,否则认为它是白色。因为我们这次全是白色,没有黑色,那么我们设置小球那一点是白色。接着卷几个向左移动一格:

 

我们发现还是全白,红球那点肯定是白色,我们就这样搞,搞到一行的末尾:

 

多出来的地方我们用黑色矩阵来填充,所以这一行结束之后,我们得到这一行的像素全是白色的。然后我们移动到第二行第一列:

 

继续运算,我们这一行也都是白色。

接下来我们就这样一直地移动,直到来到了这里:

 

我们移动到了第四行第四列之后,我们发现,这特么的完全重合了,而且底下全都是黑色的,那么这一个点就保留它的黑色啦。然后我们继续移动,直到最后一行最后一列被遍历完成。

 

左边的图就是腐蚀之前的图片,右边的图是腐蚀之后的,我们发现啥了呢?黑色的小框变少了。也就是说腐蚀掉了。

我们把黑色的小框想像成为高亮度的点,白色的小框想像成为低亮度的点,那么腐蚀后的结果是啥样呢?很自然的我们会知道,高亮度的点点全都消失了。这就是腐蚀的意思。我们把黑色的矩形想象成高亮度的噪点的话,那么我们不就是把噪声给去除了哈?

注意,这里有一个东西大家可能会懵逼,就是我们这里黑色的矩形指的是高亮的,也就是白点,白色的矩形指的是低亮度的,也就是黑点。大家别搞反了!

官方给过几个图片,来说明腐蚀,地址是:

http://www.opencv.org.cn/opencvdoc/2.3.2/html/doc/tutorials/imgproc/erosion_dilatation/erosion_dilatation.html

上面有两张图:

第一张是原图:

 

第二张是腐蚀后的图图:

 

我们发现,黑色的,也就是低亮度的像素明显增加了,而高亮度的白色相应减少了。

接下来讲一下膨胀:

膨胀

膨胀是是将灰度值大的像素扩展,主要用来去除边缘的坑,也就是去毛刺。膨胀和腐蚀相反,腐蚀是全都高亮度保留高亮度,否者置为低亮度。膨胀是,卷积核上面的所有位置有一个是高亮度,中心位置置为高亮度,否者置为低亮度。

我们还是要用左边的6*6的图和右边的卷积核做卷积。其中,图片中黑色的矩形区域表示亮度大的位置,白色的小矩形区域表示亮度小的位置。

 

首先我们把卷积核的红色小球位置移动到图片的第一行第一列,然后我们发现三个矩形的位置全部都是白色的。我们得出第一行第一列的位置是白色的。接下来向右移动一个小格,我们发现还是白色的,那么我们就设定第二个小格,我们发现还是全白,那么我们设定它为白色。接着继续移动,不久就到了这个:

 

我们发现第二行第五列是黑色,所以我们红色小球位置置为黑色。接下来继续移动,一直到了这个位置:

 

我们发现第二行第五列和第三行第四列都是黑色,所以第二行第四列置为黑色。

就这样我们一直移动,直到最后,我们就得到了膨胀后的图片:

 

左边是原图,右边的是膨胀后的图片,我们发现黑色的小矩形增加了,说白了也就是高亮度的位置增加了。也就是低亮度的边缘变小了。

我们还是用官方的图片来说明一下:

有两张图:

第一张是原图:

 

第二张是膨胀后的图图:

 

稍有常识的人都能看的出来,这个图片变细了啊,说白了就是边缘变小了。也就是图片缩小了一圈,毛刺被去除了,这个就是膨胀的作用了。

开闭运算

为了让我们的图片去除毛刺和噪声,变得更加的细腻,一般的方法是做一个先开后闭的运算。全称叫做开运算-闭运算。

开运算指的是先腐蚀一下后膨胀一下。闭运算是先膨胀一下后腐蚀一下。先开后闭运算指的就是开运算一次再闭运算一次。

程序实例

最后一部分,本人要讲解一个程序实例,首先我们有一张图:

 

我们可以清楚地看到,上面全是噪声点,我们通过程序做一个先开后闭的操作,也就是腐蚀->膨胀->膨胀->腐蚀。

膨胀操作的方法是:

void dilate(const Mat& src, Mat& dst, const Mat& element, Point anchor=Point(-1, -1), int iteratiOns=1, int borderType=BORDER_CONSTANT, const Scalar& borderValue=morphologyDefaultBorderValue())

这看起来好乱好乱,我给大家一个个地讲一下含义。

第一个参数src指的是输入的图像,也就是我们带噪声的小熊的图片。

第二个参数dst是输出的图像,也就是膨胀后得到的图片。

第三个参数是卷积核,如果不指定的话,就按照3*3的简单矩形卷积核而定。

第四个参数是中心点的位置,如果不指定,那么默认为中心,也就是卷积核的中央。

第五个参数是膨胀的次数,不指定就为1次。

第六个参数是边界常量,也就是我们卷积核超出边界了,超出部分的图片按那个值算,不指定的话按0计算。

第七个参数是是调节卷积核的,我看不太懂,所以在这里面不讲了,按默认的来了。如果大家比较懂的话,可以留言告诉我一下,第一个告诉我的人给10元大红包,

腐蚀操作的方法是:

void erode(const Mat& src, Mat& dst, const Mat& element, Point anchor=Point(-1, -1), int iteratiOns=1, int borderType=BORDER_CONSTANT, const Scalar& borderValue=morphologyDefaultBorderValue())

参数和膨胀一模一样,没啥可说的了。

至于开闭运算,我们其实不用腐蚀->膨胀->膨胀->腐蚀这样写四条语句。在OPENCV里面已经帮我们写好了。

void morphologyEx(InputArray src, OutputArray dst, int op, InputArray element, Point anchor=Point(-1,-1), int iteratiOns=1, int borderType=BORDER_CONSTANT, const Scalar& borderValue=morphologyDefaultBorderValue() )

这个就是开闭运算的方法了。

第一个参数是输入图像。

第二个参数是输出图像。

第三个参数是指定我们是开运算还是闭运算,它的参数可以写成

MORPH_OPEN - an opening operation

MORPH_CLOSE - a closing operation

MORPH_GRADIENT - a morphological gradient

MORPH_TOPHAT - top hat

MORPH_BLACKHAT - black hat

前两个就是开闭运算。

第四个参数是卷积核。

第五个元素是卷积核的中心点位置,默认是卷积核的中央。

第六个参数是计算几次,默认为一次。

第七个参数是边界常量,也就是我们卷积核超出边界了,超出部分的图片按那个值算,不指定的话按0计算。

第八个参数我不太懂,和卷积膨胀的参数一样。

这样我们的方法就介绍完毕了,但是我们用的是PYTHON,上面方法是C++的,所以还需要稍微的改动一下。最后我们写出代码为:

import cv2

import numpy as np

 

#导入图片并显示

image=cv2.imread('D:/xiaomu/dawawazao.jpg')

cv2.imshow('orgin',image)

#定义一个5*5的卷积核

kernel = np.ones((5,5))    

#膨胀操作

dilateImg=cv2.dilate(image,kernel)

cv2.imshow('dilate',dilateImg)

#腐蚀操作

erodeImg=cv2.erode(image,kernel)

cv2.imshow('erode',erodeImg)

#开运算

morphOpenImg=cv2.morphologyEx(image,cv2.MORPH_OPEN,kernel)

cv2.imshow('morphOpen',morphOpenImg)

#闭运算

morphCloseImg=cv2.morphologyEx(image,cv2.MORPH_CLOSE,kernel)

cv2.imshow('morphClose',morphCloseImg)

#先开后闭

morphOpenImg=cv2.morphologyEx(image,cv2.MORPH_OPEN,kernel)

morphOpenAndCloseImg=cv2.morphologyEx(morphOpenImg,cv2.MORPH_CLOSE,kernel)

cv2.imshow('morphOpenAndCloseImg',morphOpenAndCloseImg)

#等待结束

cv2.waitKey()

cv2.destroyAllWindows()

import numpy as np

 

#导入图片并显示

image=cv2.imread('D:/xiaomu/dawawazao.jpg')

cv2.imshow('orgin',image)

#定义一个5*5的卷积核

kernel = np.ones((5,5))    

#膨胀操作

dilateImg=cv2.dilate(image,kernel)

cv2.imshow('dilate',dilateImg)

#腐蚀操作

erodeImg=cv2.erode(image,kernel)

cv2.imshow('erode',erodeImg)

#开运算

morphOpenImg=cv2.morphologyEx(image,cv2.MORPH_OPEN,kernel)

cv2.imshow('morphOpen',morphOpenImg)

#闭运算

morphCloseImg=cv2.morphologyEx(image,cv2.MORPH_CLOSE,kernel)

cv2.imshow('morphClose',morphCloseImg)

#先开后闭

morphOpenImg=cv2.morphologyEx(image,cv2.MORPH_OPEN,kernel)

morphOpenAndCloseImg=cv2.morphologyEx(morphOpenImg,cv2.MORPH_CLOSE,kernel)

cv2.imshow('morphOpenAndCloseImg',morphOpenAndCloseImg)

#等待结束

cv2.waitKey()

cv2.destroyAllWindows()

PYTHON中,如果参数选取默认值的话,可以不写了,这是与CPP的不同之处,所以上面的代码中缺少很多的东西。

下面是结果:

 

从左到右分别是原图,膨胀,腐蚀。

 

从左到右分别是开运算,闭运算,先开后闭。

我们清楚地发现,先开后闭运算之后,图片中的小点全没了,有效地去除了噪声。可是呢,图片不清晰了,这也是缺点吧,如果想要图片更清晰,换好些的卷积核实验吧。


———————————————

如果对我的课程感兴趣的话,欢迎关注小木希望学园-微信公众号: 

mutianwei521

也可以扫描二维码哦!



推荐阅读
  • Python自动化处理:从Word文档提取内容并生成带水印的PDF
    本文介绍如何利用Python实现从特定网站下载Word文档,去除水印并添加自定义水印,最终将文档转换为PDF格式。该方法适用于批量处理和自动化需求。 ... [详细]
  • 1.如何在运行状态查看源代码?查看函数的源代码,我们通常会使用IDE来完成。比如在PyCharm中,你可以Ctrl+鼠标点击进入函数的源代码。那如果没有IDE呢?当我们想使用一个函 ... [详细]
  • 深入解析Android自定义View面试题
    本文探讨了Android Launcher开发中自定义View的重要性,并通过一道经典的面试题,帮助开发者更好地理解自定义View的实现细节。文章不仅涵盖了基础知识,还提供了实际操作建议。 ... [详细]
  • 本文详细介绍如何使用Python进行配置文件的读写操作,涵盖常见的配置文件格式(如INI、JSON、TOML和YAML),并提供具体的代码示例。 ... [详细]
  • 数据管理权威指南:《DAMA-DMBOK2 数据管理知识体系》
    本书提供了全面的数据管理职能、术语和最佳实践方法的标准行业解释,构建了数据管理的总体框架,为数据管理的发展奠定了坚实的理论基础。适合各类数据管理专业人士和相关领域的从业人员。 ... [详细]
  • DNN Community 和 Professional 版本的主要差异
    本文详细解析了 DotNetNuke (DNN) 的两种主要版本:Community 和 Professional。通过对比两者的功能和附加组件,帮助用户选择最适合其需求的版本。 ... [详细]
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • 深入理解Tornado模板系统
    本文详细介绍了Tornado框架中模板系统的使用方法。Tornado自带的轻量级、高效且灵活的模板语言位于tornado.template模块,支持嵌入Python代码片段,帮助开发者快速构建动态网页。 ... [详细]
  • PHP 5.2.5 安装与配置指南
    本文详细介绍了 PHP 5.2.5 的安装和配置步骤,帮助开发者解决常见的环境配置问题,特别是上传图片时遇到的错误。通过本教程,您可以顺利搭建并优化 PHP 运行环境。 ... [详细]
  • 本文详细介绍了Java中org.eclipse.ui.forms.widgets.ExpandableComposite类的addExpansionListener()方法,并提供了多个实际代码示例,帮助开发者更好地理解和使用该方法。这些示例来源于多个知名开源项目,具有很高的参考价值。 ... [详细]
  • 本文详细介绍了Java编程语言中的核心概念和常见面试问题,包括集合类、数据结构、线程处理、Java虚拟机(JVM)、HTTP协议以及Git操作等方面的内容。通过深入分析每个主题,帮助读者更好地理解Java的关键特性和最佳实践。 ... [详细]
  • 尽管某些细分市场如WAN优化表现不佳,但全球运营商路由器和交换机市场持续增长。根据最新研究,该市场预计在2023年达到202亿美元的规模。 ... [详细]
  • 本文深入探讨了Linux系统中网卡绑定(bonding)的七种工作模式。网卡绑定技术通过将多个物理网卡组合成一个逻辑网卡,实现网络冗余、带宽聚合和负载均衡,在生产环境中广泛应用。文章详细介绍了每种模式的特点、适用场景及配置方法。 ... [详细]
  • 2023年京东Android面试真题解析与经验分享
    本文由一位拥有6年Android开发经验的工程师撰写,详细解析了京东面试中常见的技术问题。涵盖引用传递、Handler机制、ListView优化、多线程控制及ANR处理等核心知识点。 ... [详细]
  • 从 .NET 转 Java 的自学之路:IO 流基础篇
    本文详细介绍了 Java 中的 IO 流,包括字节流和字符流的基本概念及其操作方式。探讨了如何处理不同类型的文件数据,并结合编码机制确保字符数据的正确读写。同时,文中还涵盖了装饰设计模式的应用,以及多种常见的 IO 操作实例。 ... [详细]
author-avatar
蒲小平2502897955
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有