利用OpenCV4.9制作自己的线性滤波器!

 返回:OpenCV系列文章目录(持续更新中......)

上一篇:OpenCV4.9使用 inRange 的阈值操作

下一篇 :OpenCV系列文章目录(持续更新中......)

目标

在本教程中,您将学习如何:

  • 使用 OpenCV 函数 filter2D()建您自己的线性过滤器。

理论

注意:

下面的解释属于 Bradski 和 Kaehler 的 Learning OpenCV 一书。

相关

从非常一般的意义上讲,关联是图像的每个部分与运算符(内核)之间的操作。

什么是内核?

内核本质上是一个固定大小的数值系数数组,以及该数组中的锚点,该锚点通常位于中心。

与内核的关联是如何工作的?

假设您想知道图像中特定位置的结果值。相关性的值按以下方式计算:

  1. 将内核锚点放在确定的像素上,内核的其余部分覆盖图像中相应的局部像素。
  2. 将核系数乘以相应的图像像素值,然后对结果求和。
  3. 将结果放置在输入图像中锚点的位置。
  4. 通过扫描整个图像上的内核,对所有像素重复该过程。

以等式的形式表示上述过程,我们将得到:

幸运的是,OpenCV 为您提供了 filter2D()函数,因此您不必编写所有这些操作。

这个程序是做什么的?

  • 加载图像
  • 执行规范化的框筛选器。例如,对于大小为 (size = 3) 的内核,内核为:

该程序将对大小为 3、5、7、9 和 11 的内核执行过滤器操作。

  • 筛选器输出(包括每个内核)将在 500 毫秒内显示

代码:

C++

教程代码如下行所示。

您也可以从这里下载

#include "opencv2/imgproc.hpp"
#include "opencv2/imgcodecs.hpp"
#include "opencv2/highgui.hpp"
 
using namespace cv;
 
int main ( int argc, char** argv )
{
 // Declare variables
 Mat src, dst;
 
 Mat kernel;
 Point anchor;
 double delta;
 int ddepth;
 int kernel_size;
 const char* window_name = "filter2D Demo";
 
 const char* imageName = argc >=2 ? argv[1] : "lena.jpg";
 
 // Loads an image
 src = imread( samples::findFile( imageName ), IMREAD_COLOR ); // Load an image
 
 if( src.empty() )
 {
 printf(" Error opening image\n");
 printf(" Program Arguments: [image_name -- default lena.jpg] \n");
 return EXIT_FAILURE;
 }
 
 // Initialize arguments for the filter
 anchor = Point( -1, -1 );
 delta = 0;
 ddepth = -1;
 
 // Loop - Will filter the image with different kernel sizes each 0.5 seconds
 int ind = 0;
 for(;;)
 {
 // Update kernel size for a normalized box filter
 kernel_size = 3 + 2*( ind%5 );
 kernel = Mat::ones( kernel_size, kernel_size, CV_32F )/ (float)(kernel_size*kernel_size);
 
 // Apply filter
 filter2D(src, dst, ddepth , kernel, anchor, delta, BORDER_DEFAULT );
 imshow( window_name, dst );
 
 char c = (char)waitKey(500);
 // Press 'ESC' to exit the program
 if( c == 27 )
 { break; }
 
 ind++;
 }
 
 return EXIT_SUCCESS;
}

解释

C++

加载图像

 const char* imageName = argc >=2 ? argv[1] : "lena.jpg";
 
 // Loads an image
 src = imread( samples::findFile( imageName ), IMREAD_COLOR ); // Load an image
 
 if( src.empty() )
 {
 printf(" Error opening image\n");
 printf(" Program Arguments: [image_name -- default lena.jpg] \n");
 return EXIT_FAILURE;
 }

初始化参数

 // Initialize arguments for the filter
 anchor = Point( -1, -1 );
 delta = 0;
 ddepth = -1;

循环

执行无限循环,更新内核大小,并将线性滤波器应用于输入图像。让我们更详细地分析一下:

  • 首先,我们定义过滤器将使用的内核。在这里:
 // Update kernel size for a normalized box filter
 kernel_size = 3 + 2*( ind%5 );
 kernel = Mat::ones( kernel_size, kernel_size, CV_32F )/ (float)(kernel_size*kernel_size);

第一行是将kernel_size更新为以下范围内的奇数值:[3,11]。第二行实际上是通过将其值设置为填充1's 的矩阵并通过将其除以元素数对其进行归一化来构建内核的。

  • 设置完内核后,我们可以使用函数 filter2D()生成过滤器:
 // Apply filter
 filter2D(src, dst, ddepth , kernel, anchor, delta, BORDER_DEFAULT );
  • 参数表示:
    • src:源图片
    • dst:目标图像
    • ddepthdst 的深度。负值(如 \(-1\))表示深度与源相同。
    • kernel:要通过图像扫描的内核
    • anchor:锚点相对于其内核的位置。位置 Point(-1, -1) 默认表示中心。
    • delta:在关联期间要添加到每个像素的值。默认情况下,它是 \(0\)
    • BORDER_DEFAULT:我们默认允许此值(以下教程中的更多详细信息)
  • 我们的程序将实现一个 while 循环,每 500 毫秒,我们过滤器的内核大小将在指示的范围内更新。

结果

  1. 编译上面的代码后,您可以执行它,并给出图像的路径作为参数。结果应为显示被归一化滤镜模糊的图像的窗口。每 0.5 秒,内核大小就会发生变化,如下面的一系列快照所示:

参考文献:

1:《Making your own linear filters!》---- Ana Huamán


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/561027.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Jenkins和gitlab实现CICD

1 背景 在开发TracerBackend服务的时候,每次更改代码之后需要推送到gitlab,然后ssh登录到Ubuntu的服务器上部署新的代码。服务成功启动之后,在本地执行测试用例,觉得这一套操作流程还是挺复杂的。想起公司的代码发布流程&#xf…

git工具简单使用

文章目录 git上传克隆README.gitignore常用指令冲突 git 进行版本控制的版本控制器。安装git yum install -y git 配置git git config --global user.email "youexample.com" 告诉git你的邮箱是什么?最好输入你的gitee的注册邮箱git config --global …

面向对象(封装,继承,多态)

1.封装【encapsulation】 【/ɪnˌkpsjuˈleɪʃ(ə)n/】 在面向对象程式设计方法中,封装(英语:Encapsulation)是指一种将抽象性函式接口的实现细节部分包装、隐藏起来的方法。 封装可以被认为是一个保护屏障,防止该…

Day60 单调栈 part03

Day60 单调栈 part03 最后一天啦!完结撒花~ 84.柱状图中最大的矩形 我的思路: 感觉和接雨水差不多,只需要多考虑一些情况 双指针 lheight 和 rheight 分别是用来存储每个柱子的左边界和右边界的数组。 解答: class Solutio…

vue element ui 打开弹窗出现黑框问题

文章目录 问题描述解决方案 问题描述 大家好!今天是2024年4月20日 | 农历三月十二,周六的我又做在公司里面写起了代码 今天在做项目的时候遇到一个奇怪的问题,如下图所示: 因为这个页面我做了两个弹框,先弹出来第一个弹…

前端工程化01-复习jQuery当中的AJAX

4.1、基础概念 什么是服务器 一台存储网站内容、网站文件的电脑 什么是资源 网站中使用的文件(html、css、图片、…)这些东西就叫做资源数据也是服务器上的资源,而且是一个网站的灵魂 客户端 客户端应该指上网的设备但是在前端开发中&a…

SQLite 的命令行 Shell(三十一)

返回:SQLite—系列文章目录 上一篇:SQLite FTS5 扩展(三十) 下一篇:SQLite—系列文章目录 1. 入门 SQLite 项目提供了一个名为 sqlite3(或 Windows 上的sqlite3.exe)的简单命令行程序 …

密码学 | 承诺:绑定性 + 隐藏性

🥑原文:承诺方案(Commitment)学习笔记 🥑写在前面: 本文属搬运博客,自己留存学习。本文只会讲承诺的两个安全属性,不会再讲解承诺的定义。 正文 承诺方案需要满足两个安全属性&…

【Pytorch】Yolov5中CPU转GPU过程报错完善留档归纳

Yolov5 从CPU转GPU Python多版本切换 Conda包处理 文章目录 Yolov5 从CPU转GPU Python多版本切换 Conda包处理1.Pytorch套件中存在版本不匹配2.numpy停留在3.8没跟上pytorch2.2.23.ModuleNotFoundError: No module named pandas._libs.interval4.ImportError: cannot imp…

SpringBoot启动流程深度解析

写在前面: 由于该系统是底层系统,以微服务形式对外暴露dubbo服务,所以本流程中SpringBoot不基于jetty或者tomcat等容器启动方式发布服务,而是以执行程序方式启动来发布(参考下图keepRunning方法)。 本文以调试一个实际的SpringBoo…

【做一名健康的CSDNer】

程序员由于工作性质,常常需要长时间面对电脑,这可能对身心健康带来挑战。以下是一些实用的建议,帮助程序员保持身心健康: 规律生活:建立健康的生活习惯,包括规律的作息时间和固定的饮食时间,保证…

书生·浦语大模型实战营Day04OpenXLab 部署

书生浦语大模型实战营Day04OpenXLab 部署 如何在 OpenXLab 部署一个 InternLM2-7B chat 的应用。 OpenXLab浦源平台介绍 OpenXLab 浦源平台以开源为核心,旨在构建开源开放的人工智能生态,促进学术成果的开放共享。OpenXLab面向 AI 研究员和开发者提供…

【python】使用python和selenium实现某平台自动化上传作品的全步骤

第一,我们需要下载python并安装 下载地址:https://www.python.org/downloads/release/python-3123/ 3.x版本的python自带pip工具,因此不需要额外下载。 ModuleNotFoundError: No module named seleniumpip用于下载python适用的各类模块&…

yolov8自带的P2层如何开启

YOLOv8模型 简述YOLOv8不同size模型简述 在最开始的YOLOv8提供的不同size的版本,包括n、s、m、l、x(模型规模依次增大,通过depth, width, max_channels控制大小),这些都是通过P3、P4和P5提取图片特征; 正…

网络编程 -- 简易TCP网络程序

一 字符串回响 1.1 核心功能 字符串回响程序类似于 echo 指令,客户端向服务器发送消息,服务器在收到消息后会将消息发送给客户端,该程序实现起来比较简单,同时能很好的体现 socket 套接字编程的流程。 1.2 程序结构 这个程序我们…

Python编程玩转二维码

文章目录 Python编程玩转二维码第一部分:背景介绍第二部分:qrcode库是什么?第三部分:如何安装这个库?第四部分:库函数使用方法第五部分:场景应用第六部分:常见Bug及解决方案第七部分…

动力学重构/微分方程参数拟合 - 基于模型

这一篇文章,主要是给非线性动力学,对微分方程模型参数拟合感兴趣的朋友写的。笼统的来说,这与混沌系统的预测有关;传统的机器学习的模式识别虽然也会谈论预测结果,但他们一般不会涉及连续的预测。这里我们考虑的是&…

Git ignore、exclude for TortoiseGit 小结

1.Ignore Type:忽略类型,也即忽略规则,如何去忽略文件? 1.1.Ignore item(s) only in containing folder(s):仅忽略在包含在文件夹中项目。 仅忽略该文件夹下选定的patterns。the patterns其实就是文件类型,比如.txt后…

文本美学:text-image打造视觉吸引力

当我最近浏览 GitHub 时,偶然发现了一个项目,它能够将文字、图片和视频转化为文本,我觉得非常有趣。于是我就花了一些时间了解了一下,发现它的使用也非常简单方便。今天我打算和家人们分享这个发现。 项目介绍 话不多说&#xf…

面试(05)————Redis篇

目录 一、项目中哪些地方使用了redis 问题一:发生了缓存穿透该怎么解决? 方案一:缓存空数据 方案二:布隆过滤器 模拟面试 问题二: 发生了缓存击穿该怎么解决? 方案一:互斥锁 方案二&#xff…