形态学操作+实例分析(第六天)

2024-05-28 20:18

本文主要是介绍形态学操作+实例分析(第六天),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

形态学概念介绍

形态学现在学完基本的几个了,但我还是不知道什么是形态学!原理其实就是和“卷积”在图像处理中的应用一样,就是一个“内核”遍历图像之后进行处理,内核的不同使得处理得到的图像效果也是不同的。下面介绍几种形态学滤波原理你就懂了:

注->RGB:0-255,0代表黑色,255代表白色

腐蚀:腐蚀的顾名思义就是一个东西变黑变坏变烂了,那么简单的理解就是把大于0的像素都都变得接近0就行了啊!

         那对应的图像处理:

膨胀:一个人膨胀了的样子怎样的?变得越来越耀眼、越来越明亮!那么对应的像素就是像素<255的就越来越接近255啊。

        那对应的图像处理:

开运算:从名字记忆是图像打开,既然是打开那就是最后的结果是膨胀-------------------->先腐蚀后膨胀

闭运算:和开运算相对,从名字记忆是图像关闭,既然是打开那就是最后的结果是腐蚀----->先膨胀后腐蚀

形态学梯度:梯度就是一个阶梯的长度,对应于图像那就是像素的差值,膨胀—原图、原图—腐蚀、膨胀—腐蚀、X/Y等方向的

顶帽:不解释了---->原图—开运算

黑帽:------------>原图—闭运算

---实例分析---

注意点:形态学滤波一般运用在二值化的图像上,对于那些彩色的图像运用不明显(用过之后很难看),看了很多书本的介绍都是随便找个例子,这是在课程中看到的,感觉按照下面的步骤学习形态学真的很简单而且实用!

例一:腐蚀的作用 

原图如下,去除图片上的小白点。

用内核大小3X3进行的图片:小的白点已经没有了,但是稍微大点的杂点还是没去除!

 这是实用15X15的内核进行的图片:图片的白点完全去除了。

这是不是完成了我们的要求了呢?仔细的看会发现,我们想要的大白色区域变小了,这是什么原因呢?

从我们上面的原理分析可以得知:腐蚀会把目标区域给变小的,请看下面的图片->>>红色区域是内核,

一号区域->黑色

二号区域->白色

三号区域->黑色

二号区域->黑色

五号区域->黑色

所以图片缩小的区域就是内核的大小,每个边都会缩小!

 在想一下,如果我们用膨胀处理经过腐蚀的图片会怎么样?由上面的分析可以很快得到结论,就是恢复我们目标区域的原始尺寸

    看下面的效果图:

 

 代码比较简单,就是几行API,但是如何运用,为什么这么运用,这才是关键:

 1 int main(int argc,char**argv)2 {3     Mat input_image = imread("1.jpg");4     if (input_image.data==NULL) {5         return -1; cout << "can't open image.../";6     }7     imshow("Sourse image", input_image);8     Mat output_image;9     Mat kernel = getStructuringElement(MORPH_RECT,Size(15,15));
10     erode(input_image,output_image,kernel);
11     dilate(output_image, output_image,kernel);
12     imshow("Destinate image",output_image);
13     waitKey(0);
14     return 0;
15 }

例二:提取行和列

    要求提取其中的行线段-->>

 经过处理的线段图片:

 

这里没给其他的特殊照片,看代码直接改一下就可以了。

注意点: getStructuringElement()获得内核的一些参数->核大小、核形状、核锚点等。其中控制核的大小可以滤波不同的噪点:

我要滤去下面的三个大噪点,保留上面的大白色区域,其实滤波的核定义成红色的大小就可以了,不一定是正方形,矩形就可以了。

int main(int argc,char*=*argv)2 {3     Mat input_image = imread("1.jpg");4     if (input_image.data==NULL) {5         return -1; cout << "can't open image.../";6     }7     imshow("Sourse image", input_image);8     Mat output_image;9     Mat kernel = getStructuringElement(MORPH_RECT,Size(input_image.cols/30,1));//这个input_image.cols/30,是定义核的长度是图片长度的三十分之一,如果直                                                           接给定一个数200也可以,但是你不知道200在图像上是多大啊。
10     erode(input_image,output_image,kernel);
11     dilate(output_image, output_image,kernel);
12     imshow("Destinate image",output_image);
13     waitKey(0);
14     return 0;
15 }

 例三:简单的提取字母

       目的是提取图片中的字母

 
灰度化:

 阈值化:

 形态学滤波:

取反之后:

 1 int main(int argc,char**argv)2 {3     Mat input_image = imread("1.jpg");4     if (input_image.data==NULL) {5         return -1; cout << "can't open image.../";6     }7     imshow("Sourse image", input_image);8     cvtColor(input_image,input_image,CV_RGB2GRAY);9     imshow("Sourse1 image", input_image);
10     //adaptiveThreshold(input_image,input_image,255,ADAPTIVE_THRESH_GAUSSIAN_C,THRESH_BINARY_INV,171,0);
11     threshold(input_image,input_image,15,255,THRESH_BINARY);
12     imshow("Sourse2 image", input_image);
13     Mat output_image;
14     Mat kernel = getStructuringElement(MORPH_RECT,Size(3,3));
15      erode(input_image,output_image,kernel);
16     dilate(output_image, output_image,kernel);
17     imshow("Destinate image",output_image);
18     bitwise_not(output_image,output_image);
19     imshow("Destinate2 image", output_image);
20     Mat my_kernel = (Mat_<uchar>(3, 3) << 0, -1, 0, 5, -1, 0, -1, 0);
21     filter2D(output_image,output_image,output_image.depth(),my_kernel);//加强显示
22     imshow("Destinate3 image", output_image);
23     waitKey(0);
24     return 0;
25 }

 例四:稍微困难的提取字母

                这个图像对我来说有点麻烦的,形态学滤波不行的,而且形态学操作之后留下很多噪点。。。。。

形态学操作之后:

轮廓检测去除噪点:

霍夫变换去除粗实线:

这个图的小噪点用上面的步骤可以去除,这个就没再继续了

上代码:

 1 Mat input_image = imread("2.jpg");2     if (input_image.data==NULL) {3         return -1; cout << "can't open image.../";4     }5     imshow("Sourse image", input_image);6     cvtColor(input_image,input_image,CV_RGB2GRAY);7     //adaptiveThreshold(input_image,input_image,255,ADAPTIVE_THRESH_GAUSSIAN_C,THRESH_BINARY_INV,171,0);8     threshold(input_image,input_image,0,255,THRESH_BINARY|THRESH_OTSU);9     //-----------------------去除细实线------------------------//
10     Mat output_image;
11     Mat kernel1 = getStructuringElement(MORPH_RECT,Size(3,3));
12     morphologyEx(input_image, input_image, MORPH_CLOSE, kernel1);
13     bitwise_not(input_image, input_image);
14     output_image = input_image.clone();
15     imshow("DeleteThick image", input_image);
16     //----------------------去除形态学不能去除的噪点----------------------//
17     vector<vector<Point> > contours;
18     vector<Vec4i> hierarchy;
19     findContours(output_image, contours, hierarchy, RETR_TREE, CHAIN_APPROX_SIMPLE);
20     Mat contours_image = Mat::zeros(input_image.size(), input_image.type());
21     for (size_t i = 0; i < contours.size(); i++)
22     {
23         double Area = contourArea(contours[i]);
24         if (Area > 50) continue;
25         drawContours(contours_image, contours, static_cast<int>(i),Scalar(255,255,255),1);
26     }
27     input_image = input_image - contours_image;
28     morphologyEx(input_image, input_image, MORPH_OPEN, kernel1);
29     imshow("contours image", input_image);
30     //------------------去除粗实线--------------------//
31     vector<Vec4i> lines;
32     HoughLinesP(input_image,lines,1,CV_PI/180,100,0,200); 
33     Mat line_image = Mat::zeros(input_image.size(), input_image.type());
34     for (size_t i = 0; i < lines.size(); i++)
35     {
36         line(line_image, Point(lines[i][0], lines[i][1]),
37             Point(lines[i][2], lines[i][3]), Scalar(255, 255, 255), 1, 8);
38     }
39     bitwise_not(input_image, input_image);
40     input_image = line_image + input_image;
41     morphologyEx(input_image, input_image, MORPH_CLOSE, kernel1);
42     imshow("Last image", input_image);

这篇关于形态学操作+实例分析(第六天)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

怎样通过分析GC日志来定位Java进程的内存问题

《怎样通过分析GC日志来定位Java进程的内存问题》:本文主要介绍怎样通过分析GC日志来定位Java进程的内存问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、GC 日志基础配置1. 启用详细 GC 日志2. 不同收集器的日志格式二、关键指标与分析维度1.

Python实现对阿里云OSS对象存储的操作详解

《Python实现对阿里云OSS对象存储的操作详解》这篇文章主要为大家详细介绍了Python实现对阿里云OSS对象存储的操作相关知识,包括连接,上传,下载,列举等功能,感兴趣的小伙伴可以了解下... 目录一、直接使用代码二、详细使用1. 环境准备2. 初始化配置3. bucket配置创建4. 文件上传到os

mysql表操作与查询功能详解

《mysql表操作与查询功能详解》本文系统讲解MySQL表操作与查询,涵盖创建、修改、复制表语法,基本查询结构及WHERE、GROUPBY等子句,本文结合实例代码给大家介绍的非常详细,感兴趣的朋友跟随... 目录01.表的操作1.1表操作概览1.2创建表1.3修改表1.4复制表02.基本查询操作2.1 SE

MySQL中的表连接原理分析

《MySQL中的表连接原理分析》:本文主要介绍MySQL中的表连接原理分析,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、背景2、环境3、表连接原理【1】驱动表和被驱动表【2】内连接【3】外连接【4编程】嵌套循环连接【5】join buffer4、总结1、背景

c++中的set容器介绍及操作大全

《c++中的set容器介绍及操作大全》:本文主要介绍c++中的set容器介绍及操作大全,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录​​一、核心特性​​️ ​​二、基本操作​​​​1. 初始化与赋值​​​​2. 增删查操作​​​​3. 遍历方

java向微信服务号发送消息的完整步骤实例

《java向微信服务号发送消息的完整步骤实例》:本文主要介绍java向微信服务号发送消息的相关资料,包括申请测试号获取appID/appsecret、关注公众号获取openID、配置消息模板及代码... 目录步骤1. 申请测试系统2. 公众号账号信息3. 关注测试号二维码4. 消息模板接口5. Java测试

MySQL数据库的内嵌函数和联合查询实例代码

《MySQL数据库的内嵌函数和联合查询实例代码》联合查询是一种将多个查询结果组合在一起的方法,通常使用UNION、UNIONALL、INTERSECT和EXCEPT关键字,下面:本文主要介绍MyS... 目录一.数据库的内嵌函数1.1聚合函数COUNT([DISTINCT] expr)SUM([DISTIN

MySQL追踪数据库表更新操作来源的全面指南

《MySQL追踪数据库表更新操作来源的全面指南》本文将以一个具体问题为例,如何监测哪个IP来源对数据库表statistics_test进行了UPDATE操作,文内探讨了多种方法,并提供了详细的代码... 目录引言1. 为什么需要监控数据库更新操作2. 方法1:启用数据库审计日志(1)mysql/mariad

springboot如何通过http动态操作xxl-job任务

《springboot如何通过http动态操作xxl-job任务》:本文主要介绍springboot如何通过http动态操作xxl-job任务的问题,具有很好的参考价值,希望对大家有所帮助,如有错... 目录springboot通过http动态操作xxl-job任务一、maven依赖二、配置文件三、xxl-

python中Hash使用场景分析

《python中Hash使用场景分析》Python的hash()函数用于获取对象哈希值,常用于字典和集合,不可变类型可哈希,可变类型不可,常见算法包括除法、乘法、平方取中和随机数哈希,各有优缺点,需根... 目录python中的 Hash除法哈希算法乘法哈希算法平方取中法随机数哈希算法小结在Python中,