6.8方框滤波

基本概念

方框滤波(Box Filter)是一种基本的图像处理技术,用于对图像进行平滑处理模糊效果。它通过在图像上应用一个固定大小的方框核(通常是矩形),计算该区域内像素值的平均值来替换中心像素的值。这种方式简单且计算效率高,但在处理边界时需要特别注意。

方框滤波(Box Filter)是一种简单的线性滤波器,它可以用于平滑图像或降低噪声。方框滤波器使用一个均匀加权的矩形核来对图像进行卷积,从而实现图像的平滑处理。在OpenCV中,方框滤波可以使用 boxFilter 函数来实现。

函数原型
在OpenCV中,方框滤波可以通过boxFilter函数实现。函数原型如下:

void boxFilter(InputArray src, OutputArray dst, int ddepth, Size ksize, Point anchor=Point(-1,-1), bool normalize=true, int borderType=BORDER_DEFAULT);
参数说明
src: 输入图像,可以是单通道或多通道图像。
dst: 输出图像,将具有与输入图像相同的尺寸,但深度取决于ddepth参数。
ddepth: 输出图像的深度。常见的选择有:-1: 输出图像深度与输入图像相同。CV_8U: 8位无符号整数。CV_16S: 16位有符号整数。CV_32F: 32位浮点数。CV_64F: 64位浮点数。
ksize: 方框核的大小,指定为一个Size对象,例如Size(3, 3)。
anchor: 核的锚点位置,默认为Point(-1,-1),表示锚点在核的中心。
normalize: 是否归一化。如果设置为true(默认值),则输出的每个像素将是核内所有像素值的平均值;如果设置为false,则输出的每个像素将是核内所有像素值的总和。
borderType: 边界处理类型。常见的边界处理方式有:BORDER_CONSTANT: 使用常数值填充边界外区域。BORDER_REPLICATE: 复制边界像素。BORDER_REFLECT: 镜像反射边界。BORDER_WRAP: 边界环绕(类似于纹理坐标)。BORDER_REFLECT_101 或 BORDER_DEFAULT: 默认的边界反射方式。

使用示例1

下面是一个详细的示例,展示如何使用boxFilter函数来进行图像平滑处理。

步骤一:包含必要的头文件
#include <opencv2/opencv.hpp>
#include <iostream>
using namespace cv;
using namespace std;步骤二:加载图像
int main(int argc, char** argv)
{Mat src = imread("path_to_your_image.jpg", IMREAD_COLOR);if (src.empty()){cout << "Error: Image cannot be loaded!" << endl;return -1;}步骤三:定义输出图像Mat dst;步骤四:应用方框滤波// 设置方框核的大小Size ksize(5, 5);  // 5x5的核// 应用方框滤波boxFilter(src, dst, -1, ksize, Point(-1,-1), true);
在这里,-1表示输出图像的深度与输入图像相同。true表示启用归一化,即输出的每个像素值将是核内所有像素值的平均值。步骤五:显示结果namedWindow("Original Image", WINDOW_AUTOSIZE);imshow("Original Image", src);namedWindow("Blurred Image", WINDOW_AUTOSIZE);imshow("Blurred Image", dst);waitKey(0);return 0;
}注意事项
归一化: 如果normalize参数设置为true,则输出图像中的每个像素值将是核内所有像素值的平均值。如果设置为false,则输出图像中的每个像素值将是核内所有像素值的总和。在某些情况下,不归一化的结果可能会导致像素值溢出。
边界处理: 方框滤波在处理图像边界时可能会出现问题,特别是当核的大小超过边界时。选择适当的边界处理方式可以帮助解决这些问题。
性能: 方框滤波是一种简单快速的滤波方法,但由于它在每个像素上都执行了累加操作,因此在处理大尺寸核时可能会消耗较多资源。对于大型核,可以考虑使用积分图(integral image)来加速计算。通过上述步骤,你可以使用OpenCV的boxFilter函数来对图像进行平滑处理或实现模糊效果。

#include <opencv2/opencv.hpp>
#include <iostream>
using namespace cv;
using namespace std;int main(int argc, char** argv)
{Mat src = imread("880.jpeg", IMREAD_COLOR);if (src.empty()){cout << "Error: Image cannot be loaded!" << endl;return -1;}Mat dst;// 设置方框核的大小Size ksize(5, 5);  // 5x5的核// 应用方框滤波boxFilter(src, dst, -1, ksize, Point(-1, -1), true);namedWindow("Original Image", WINDOW_NORMAL);imshow("Original Image", src);namedWindow("Blurred Image", WINDOW_NORMAL);imshow("Blurred Image", dst);waitKey(0);return 0;
}注意事项
归一化: 如果normalize参数设置为true,则输出图像中的每个像素值将是核内所有像素值的平均值。如果设置为false,则输出图像中的每个像素值将是核内所有像素值的总和。在某些情况下,不归一化的结果可能会导致像素值溢出。
边界处理: 方框滤波在处理图像边界时可能会出现问题,特别是当核的大小超过边界时。选择适当的边界处理方式可以帮助解决这些问题。
性能: 方框滤波是一种简单快速的滤波方法,但由于它在每个像素上都执行了累加操作,因此在处理大尺寸核时可能会消耗较多资源。对于大型核,可以考虑使用积分图(integral image)来加速计算。通过上述步骤,你可以使用OpenCV的boxFilter函数来对图像进行平滑处理或实现模糊效果。

运行结果1

示例代码2

示例代码下面是一个使用OpenCV C++实现方框滤波的示例代码:#include <iostream>
#include <opencv2/opencv.hpp>using namespace std;
using namespace cv;void applyBoxFilter(const Mat &src, Mat &dst, Size ksize, bool normalize) 
{boxFilter(src, dst, -1, ksize, Point(-1, -1), normalize);
}int main(int argc, char** argv) 
{/*if (argc != 2) {cout << "Usage: ./BoxFilter <Image Path>" << endl;return -1;}*/// 加载图像Mat img = imread("559.jpg", IMREAD_GRAYSCALE);if (!img.data) {cout << "Error opening image" << endl;return -1;}// 定义核大小Size ksize(5, 5);  // 核大小// 初始化输出矩阵Mat filtered;// 应用方框滤波applyBoxFilter(img, filtered, ksize, true);// 显示结果namedWindow("Original Image", WINDOW_NORMAL);imshow("Original Image", img);namedWindow("Filtered Image", WINDOW_NORMAL);imshow("Filtered Image", filtered);waitKey(0);destroyAllWindows();return 0;
}代码解释
1. 加载图像:使用 imread 函数加载图像。
2. 定义核大小:设置方框滤波器的核大小。
3. 初始化输出矩阵:创建一个新的矩阵来存储滤波后的图像。
4. 应用方框滤波:使用 boxFilter 函数应用方框滤波。
5. 显示结果:使用 imshow 函数显示原始图像和滤波后的图像。

运行结果2

方框滤波的应用

方框滤波器常用于以下场景:

•图像平滑:通过平均相邻像素值来减少图像中的噪声。

•均值滤波:当 normalize 参数为 true 时,方框滤波器相当于均值滤波器。

•非归一化的方框滤波:当 normalize 参数为 false 时,可以用于特殊的图像处理任务,例如累积求和。

非归一化的方框滤波示例

非归一化的方框滤波不将权重归一化,这意味着每个像素的权重保持不变。这在某些特殊情况下是有用的,例如累积求和(Integral Images)。

#include <opencv2/opencv.hpp>
#include <iostream>
using namespace cv;
using namespace std;void applyNonNormalizedBoxFilter(const Mat &src, Mat &dst, Size ksize)
{boxFilter(src, dst, -1, ksize, Point(-1, -1), false);
}int main(int argc, char** argv)
{//if (argc != 2)//{//	cout << "Usage: ./NonNormalizedBoxFilter <Image Path>" << endl;//	return -1;//}// 加载图像Mat img = imread("559.jpg", IMREAD_GRAYSCALE);if (!img.data){cout << "Error opening image" << endl;return -1;}// 定义核大小Size ksize(5, 5);  // 核大小// 初始化输出矩阵Mat filtered;// 应用非归一化的方框滤波applyNonNormalizedBoxFilter(img, filtered, ksize);// 显示结果namedWindow("Original Image", WINDOW_NORMAL);imshow("Original Image", img);namedWindow("Non-Normalized Filtered Image", WINDOW_NORMAL);imshow("Non-Normalized Filtered Image", filtered);waitKey(0);destroyAllWindows();return 0;
}

运行结果3

性能优化

方框滤波器的计算可以通过积分图像(Integral Images)来加速。积分图像是一种预先计算的数据结构,可以高效地计算任意矩形区域的和。OpenCV提供了 integral 函数来计算积分图像。

积分图像示例

积分图像可以用来高效地实现方框滤波。

#include <opencv2/opencv.hpp>
#include <iostream>
using namespace cv;
using namespace std;void applyBoxFilterUsingIntegral(const Mat &src, Mat &dst, Size ksize) 
{Mat integralImg;integral(src, integralImg);int radius = ksize.width / 2;for (int y = radius; y < src.rows - radius; ++y) {for (int x = radius; x < src.cols - radius; ++x) {int top_left = integralImg.at<int>(y - radius, x - radius);int top_right = integralImg.at<int>(y - radius, x + radius);int bottom_left = integralImg.at<int>(y + radius, x - radius);int bottom_right = integralImg.at<int>(y + radius, x + radius);int sum = bottom_right + top_left - top_right - bottom_left;dst.at<uchar>(y, x) = static_cast<uchar>(sum / (ksize.width * ksize.height));}}
}int main(int argc, char** argv) 
{if (argc != 2) {cout << "Usage: ./BoxFilterUsingIntegral <Image Path>" << endl;return -1;}// 加载图像Mat img = imread(argv[1], IMREAD_GRAYSCALE);if (!img.data){cout << "Error opening image" << endl;return -1;}// 定义核大小Size ksize(5, 5);  // 核大小// 初始化输出矩阵Mat filtered = Mat::zeros(img.size(), img.type());// 应用方框滤波applyBoxFilterUsingIntegral(img, filtered, ksize);// 显示结果imshow("Original Image", img);imshow("Filtered Image Using Integral", filtered);waitKey(0);destroyAllWindows();return 0;
}

运行结果4

总结

通过这些示例,你应该能够理解如何在OpenCV中使用C++实现方框滤波。方框滤波器是一个简单但有效的工具,可用于图像平滑和噪声减少。

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Electron 隐藏顶部菜单

隐藏前&#xff1a; 隐藏后&#xff1a; 具体设置代码&#xff1a; 在 main.js 中加入这行即可&#xff1a; // 导入模块 const { app, BrowserWindow ,Menu } require(electron) const path require(path)// 创建主窗口 const createWindow () > {const mainWindow ne…

Mybatis详细教程 (万字详解)

Mybatis 3.5.14 来自于B站‘天气预报’,一名宝藏up,跟着他可以培养起独立解决编程问题的能力&#xff01;&#xff01;&#xff01; 01.简介 1.1 官网 官方中文网: MyBatis中文网 中文网参考手册 1.2 概念 MyBatis 是一款优秀的持久层框架&#xff0c;支持自定义 SQL, 存储过…

ZLMediaKit快速上手【保姆级简单快速版】

一、前言 1、ZLMediaKit使用场景 最近在写一个摄像头检测的项目&#xff0c;其中需要做拉流测试&#xff0c;但是摄像头数量不够用&#xff0c;如果直接重复拉流可能会出现问题&#xff0c;使用ZLMediaKit&#xff08;一个基于C11的高性能运营级流媒体服务框架&#xff09;可…

UE5: Content browser工具编写02

DebugHeader.h 中的全局变量&#xff0c;已经在一个cpp file中被include了&#xff0c;如果在另一个cpp file中再include它&#xff0c;就会有一些conflicts。先全部给加一个static Add static keyword to debug functionsWrap all the functions inside of a namespaceprint …

【Redis技能熟练掌握之十年内功】

Redis技能熟练掌握之十年内功 1.redis是什么&#xff1f;为什么要使用redis&#xff1f;2.redis一般应用于什么场景&#xff08;四个场景&#xff09;&#xff1f;3. Redis持久化机制是什么&#xff1f;各自的优缺点&#xff1f;一般咋么用&#xff1f;4. redis五个基础类型支持…

【开源免费】基于SpringBoot+Vue.JS服装销售平台(JAVA毕业设计)

博主说明&#xff1a;本文项目编号 T 054 &#xff0c;文末自助获取源码 \color{red}{T054&#xff0c;文末自助获取源码} T054&#xff0c;文末自助获取源码 目录 一、系统介绍二、演示录屏三、启动教程四、功能截图五、文案资料5.1 选题背景5.2 国内外研究现状5.3 可行性分析…

docker -私有镜像仓库 - harbor安装

文章目录 1、镜像仓库简介2、Harbor简介3、下载与安装3.1、下载3.2、安装3.2.1、上传harbor-offline-installer-v2.8.2.tgz到虚拟机中解压并修改配置文件3.2.2、解压tgz包3.2.3、切换到解压缩后的目录下3.2.4、准备配置文件3.2.5、修改配置文件 4、启动Harbor5、启动关闭命令6、…

gitlab添加CI自动测试

1. CI运行规格配置文件.gitlab-ci.yml yml 配置文件语法&#xff1a; https://docs.gitlab.com/17.3/ee/ci/yaml/index.html 添加.gitlab-ci.yml文件并配置 &#xff1a; ## 定义几个阶段 stages: # List of stages for jobs, and their order of execution- buil…

982. 按位与为零的三元组

1. 题目 982. 按位与为零的三元组 2. 解题思路 随机选择两个数&#xff0c;记录两个数的与结果。以及它的次数。 然后再遍历数组&#xff0c;用第三个数去与前两个数的结果&#xff0c;如果等于0&#xff0c;则满足条件。 3. 代码 3.1. 注意点 首先用简单的思路切入&…

【顺序表使用练习】发牌游戏

【顺序表使用练习】发牌游戏 1. 介绍游戏2. 实现52张牌3. 实现洗牌4. 实现发牌5. 效果展示 1. 介绍游戏 首先先为大家介绍一下设计要求 实现52张牌&#xff08;这里排除大小王&#xff09;洗牌——打乱牌的顺序发牌——3个人&#xff0c;1人5张牌 2. 实现52张牌 创建Code对象创…

MMD模型及动作一键完美导入UE5-IVP5U插件方案(二)

1、下载并启用IVP5U插件 1、下载IVP5U插件, IVP5U,点击Latest下载对应引擎版本,将插件放到Plugins目录,同时将.uplugin文件的EnableByDefault改为false 2、然后通过Edit->Plugins启用插件 2、导入pmx模型 1、直接在Content的某个目录拖入pmx模型,选择默认参数 2、…

项目实战:k8s部署考试系统

一、新建nfs服务器&#xff08;192.168.1.44&#xff09; 1.基础配置&#xff08;IP地址防火墙等&#xff09; 2.配置时间同步 [rootlocalhost ~]# yum -y install ntpdate.x86_64 [rootlocalhost ~]# ntpdate time2.aliyun.com 27 Sep 10:28:08 ntpdate[1634]: adjust tim…

【巅峰算力,静谧之作】4卡4090GPU深度学习“静音”服务器

各位同仁&#xff0c;随着人工智能浪潮的汹涌澎湃&#xff0c;我们正步入一个前所未有的创新纪元。在这个充满挑战与机遇的时代&#xff0c;我愈发频繁地在工作场景中邂逅那些致力于深度学习探索的智者们。他们&#xff0c;对计算力的渴望如同对知识的追求一般&#xff0c;永无…

React表单:formik、final-form和react-hook-form

表单无处不在&#xff0c;它是每个网站的必备部分。在用React构建web应用时&#xff0c;处理表单是不可避免的。 你可以选择自己的方式来处理&#xff0c;或者选择社区中现成的库。然而&#xff0c;当你选择一个第三方库时&#xff0c;你会立即面临一个问题&#xff1a;有太多的…

Spring Boot 学习之路 -- 配置项目

前言 最近因为业务需要&#xff0c;被拉去研究后端的项目&#xff0c;代码框架基于 Spring Boot&#xff0c;对我来说完全小白&#xff0c;需要重新学习研究…出于个人习惯&#xff0c;会以 Blog 文章的方式做一些记录&#xff0c;文章内容基本来源于「 Spring Boot 从入门到精…

SpringMVC5-域对象共享数据

目录 使用ServletAPI向request域对象共享数据 使用ModelAndView向request域对象共享数据 使用Model向request域对象共享数据 使用map向request域对象共享数据 使用ModelMap向request域对象共享数据 Model、ModelMap、Map的关系 向session域共享数据 向application域共享…

SQLite3模块使用详解

目录 一、引言 1.1 SQLite3 简介 1.2 Python sqlite3 模块 二、连接数据库 2.1 导入 sqlite3 模块 2.2 连接数据库 2.3 创建游标对象 三、执行 SQL 语句 3.1 创建表 3.2 插入数据 3.3 查询数据 3.4 更新数据 3.5 删除数据 四、处理查询结果 4.1 fetchall() 4.2…

探探Java与python中的闭包

说在前面&#xff1a;在计算机科学中&#xff0c;闭包是指一个函数以及其引用的周围环境&#xff08;变量&#xff09;所组成的整体。简单来说&#xff0c;闭包允许一个函数访问并操作其外部函数作用域中的变量&#xff0c;即使外部函数已经执行完毕。 Java函数式编程—闭包&am…

C++map与set

文章目录 前言一、map和set基础知识二、set与map使用示例1.set去重操作2.map字典统计 总结 前言 本章主要介绍map和set的基本知识与用法。 一、map和set基础知识 map与set属于STL的一部分&#xff0c;他们底层都是是同红黑树来实现的。 ①set常见用途是去重 &#xff0c;set不…

【Java】包装类【主线学习笔记】

文章目录 前言包装类基本数据类型与包装类之间的转换基本数据类型转换为包装类可以通过以下几种方式&#xff1a;包装类转换为基本数据类型可以通过以下几种方式&#xff1a;初始化值不同与String之间的转换 前言 Java是一门功能强大且广泛应用的编程语言&#xff0c;具有跨平台…