OpenCV-基于累计直方图的中值滤波算法

作者:翟天保Steven
版权声明:著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处

实现原理

       基于累计直方图的中值滤波算法是一种图像处理技术,用于去除图像中的噪声。它利用了像素值的频数分布,通过计算累计直方图来快速确定中值,以实现滤波操作。

       下面是该算法的基本步骤:

  1. 计算累计直方图:对输入图像进行直方图统计,并计算每个像素值及其之前所有像素值的累计频数。

  2. 确定中值位置:根据累计直方图,找到频数超过总像素数一半的像素值,该值即为中值。

  3. 应用中值滤波:对图像中的每个像素,使用中值替换其原始值,以完成滤波操作。

       这种方法相对于传统的中值滤波有一定的优势,因为它利用了累计直方图的性质,使得中值的计算速度更快,尤其适用于处理灰度图像中像素值范围在0到255之间的情况。

功能函数代码

// 基于累计直方图的中值滤波算法
cv::Mat filterMedian_CPU(cv::Mat input, int filterWindowSize)
{
	int row = input.rows;
	int col = input.cols;
	int r = filterWindowSize / 2;
	cv::Mat output(row, col, CV_8UC1);
#pragma omp parallel for
	for (int i = 0; i < row; ++i)
	{
		for (int j = 0; j < col; ++j)
		{
			int ms = std::max(i - r, 0);
			int me = std::min(i + r, row - 1);
			int ns = std::max(j - r, 0);
			int ne = std::min(j + r, col - 1);
			// 初始化
			int count = 0;
			int histogram[256] = { 0 };
			// 直方图累计求中值
			for (int m = ms; m <= me; ++m)
			{
				for (int n = ns; n <= ne; ++n)
				{
					histogram[input.at<uchar>(m, n)]++;
					count++;
				}
			}
			int count_ = 0;
			int thresh = count / 2 + 1;
			int mid = 0;
			for (int k = 0; k < 256; ++k)
			{
				count_ += histogram[k];
				if (count_ >= thresh)
				{
					mid = k;
					break;
				}
			}
			output.at<uchar>(i, j) = mid;
		}
	}
	return output;
}

C++测试代码

       为了对比算法效果,我写了一个基于排序的中值滤波算法,两者均采用OpenMP并行技术提速,进行对比测试。

#include <time.h>
#include <omp.h>
#include <vector>
#include <iostream>
#include <algorithm>
#include <opencv2/opencv.hpp>

using namespace std;

// 中值滤波-CPU-排序法
cv::Mat filterMedian_CPU_Sort(cv::Mat input, int filterWindowSize)
{
	int row = input.rows;
	int col = input.cols;
	cv::Mat output(row, col, CV_8UC1);
	int r = filterWindowSize / 2;
#pragma omp parallel for
	for (int i = 0; i < row; ++i)
	{
		vector<float> datas;
		for (int j = 0; j < col; ++j)
		{
			int ms = std::max(i - r, 0);
			int me = std::min(i + r, row - 1);
			int ns = std::max(j - r, 0);
			int ne = std::min(j + r, col - 1);
			// 求窗口内有效数据的中值
			datas.clear();
			for (int m = ms; m <= me; ++m)
			{
				for (int n = ns; n <= ne; ++n)
				{
					datas.push_back(input.at<uchar>(m, n));
				}
			}
			sort(datas.begin(), datas.end());
			output.at<uchar>(i, j) = datas[datas.size() / 2];
		}
	}
	return output;
}

// 基于累计直方图的中值滤波算法
cv::Mat filterMedian_CPU(cv::Mat input, int filterWindowSize)
{
	int row = input.rows;
	int col = input.cols;
	int r = filterWindowSize / 2;
	cv::Mat output(row, col, CV_8UC1);
#pragma omp parallel for
	for (int i = 0; i < row; ++i)
	{
		for (int j = 0; j < col; ++j)
		{
			int ms = std::max(i - r, 0);
			int me = std::min(i + r, row - 1);
			int ns = std::max(j - r, 0);
			int ne = std::min(j + r, col - 1);
			// 初始化
			int count = 0;
			int histogram[256] = { 0 };
			// 直方图累计求中值
			for (int m = ms; m <= me; ++m)
			{
				for (int n = ns; n <= ne; ++n)
				{
					histogram[input.at<uchar>(m, n)]++;
					count++;
				}
			}
			int count_ = 0;
			int thresh = count / 2 + 1;
			int mid = 0;
			for (int k = 0; k < 256; ++k)
			{
				count_ += histogram[k];
				if (count_ >= thresh)
				{
					mid = k;
					break;
				}
			}
			output.at<uchar>(i, j) = mid;
		}
	}
	return output;
}
int main(void)
{
	// 测试
	cv::Mat input = cv::imread("test.jpg", 0);
	cv::Mat output0, output1;
	int filterWindowSize = 15;
	cout << "filterWindowSize:" << filterWindowSize << endl;
	// 在CPU上执行中值滤波-排序法
	clock_t ss, es;
	ss = clock();
	output0 = filterMedian_CPU_Sort(input, filterWindowSize);
	es = clock();
	cout << "Sort CPU time:" << double(es - ss) / 1000 << endl;
	// 在CPU上执行中值滤波-累计直方图法
	clock_t sc, ec;
	sc = clock();
	output1 = filterMedian_CPU(input, filterWindowSize);
	ec = clock();
	cout << "His CPU time:" << double(ec - sc) / 1000 << endl;
	return 0;
}

测试效果 

       如上图所示,分别是原图、排序法结果和直方图法结果,在速度方面,对1920*1080的图像,在窗口尺寸为15*15时,我的电脑运行速度分别是4s和0.16s。直方图算法速度快,但也有其局限性,若进行滤波的数据是非uchar型,而是更大范围甚至小数点数据,如int、double、float这种,就无法进行准确滤波,需要进一步改进才可。

       彩色图和灰度图本质上一样,无非是对三个通道进行滤波,本文就不展开了。

       如果函数有什么可以改进完善的地方,非常欢迎大家指出,一同进步何乐而不为呢~

       如果文章帮助到你了,可以点个赞让我知道,我会很快乐~加油!

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

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

相关文章

LeetCode/NowCoder-链表经典算法OJ练习2

最好的&#xff0c;不一定是最合适的&#xff1b;最合适的&#xff0c;才是真正最好的。&#x1f493;&#x1f493;&#x1f493; 目录 说在前面 题目一&#xff1a;分割链表 题目二&#xff1a;环形链表的约瑟夫问题 SUMUP结尾 说在前面 dear朋友们大家好&#xff01;&…

卷积神经网络边缘识别

为什卷积神经网络能够识别图片呢&#xff1f;是基于图片相似度比较&#xff0c;两张图片的点击越大说明两张图片越像&#xff0c;比如我们那狗胡子的图片去比较&#xff0c;如果相似度很高&#xff0c;就是认为这个动物更像狗。点积越大&#xff0c;图片越相似&#xff0c;这个…

Windows2016系统禁止关闭系统自动更新教程

目录 1.输入cmd--适合系统2016版本2.输入sconfig&#xff0c;然后按回车键3.输入5&#xff0c;然后按回车键4.示例需要设置为手动更新&#xff0c;即输入M&#xff0c;然后按回车键 1.输入cmd–适合系统2016版本 2.输入sconfig&#xff0c;然后按回车键 3.输入5&#xff0c;然后…

揭秘APP广告变现:轻松赚取收益的秘密武器,你还在等什么?

在移动互联网时代&#xff0c;APP广告变现已成为许多开发者和公司获取收益的重要方式。它如同一把秘密武器&#xff0c;帮助那些掌握了其使用技巧的人轻松赚取收益。那么&#xff0c;究竟什么是APP广告变现&#xff1f;又如何通过它轻松赚取收益呢&#xff1f;接下来&#xff0…

对中介者模式的理解

目录 一、场景1、题目 【[来源](https://kamacoder.com/problempage.php?pid1094)】1.1 题目描述1.2 输入描述1.3 输出描述1.4 输入示例1.5 输出示例 二、不采用中介者设计模式1 代码2 问题 三、中介者设计模式1 代码2 更好的例子 四、个人思考 一、场景 设计模式不是银弹&am…

STM32(开篇总结)

STM32介绍 STM32是意法半导体公司基于ARM Cortex-M内核开发的32位微控制器 STM32常应用在嵌入式领域&#xff0c;如智能车、无人机、机器人、无线通信、物联网、工业控制、娱乐电子产品等 STM32功能强大、性能优异片上资源丰富、功耗低&#xff0c;是一款经典的嵌入式微控制器…

Lesson5--二叉树(超详细版)

【本节目标】 1. 树概念及结构 2. 二叉树概念及结构 3. 二叉树顺序结构及实现 4. 二叉树链式结构及实现 1.树概念及结构 1.1树的概念 树是一种 非线性&#xff08;线性结构就是顺序表链表&#xff09; 的数据结构&#xff0c;它是由 n &#xff08; n>0 &#xff09;个…

超详细比特币Brc-20部署发布:实用步骤演示,请点赞收藏!(一)

大家好&#xff0c;我是程序员大猩猩。 上一节我们讲到如何使用Remix部署智能合约到币安链&#xff0c;如下&#xff1a; 使用Remix部署智能合约到币安链&#xff08;Remix的操作介绍 币安链合约的部署&#xff09; 有很多小伙伴私信问我&#xff0c;那么BTC&#xff08;比特…

linux系统——文件系统挂载原理

linux中从根目录到下面文件&#xff0c;用一套文件系统&#xff0c;当插入外来磁盘&#xff0c;如u盘时&#xff0c;两套文件系统如何进行交互&#xff1f; 挂载即为将一个存储设备连接到另一个已经存在的文件夹中&#xff0c;访问这个文件夹&#xff0c;即为访问该设备存储内容…

如何将 DFMini player MP3 模块与 Arduino 结合使用

要创建此项目&#xff0c;您将使用&#xff1a; DFPlayer迷你MP3模块 10kΩ电阻 开关按钮 面包板 Arduino UNO 杜邦线 现在&#xff0c;我们将学习如何构建该项目。 什么是DF Mini Player MP3模块 DFMini Player 模块是一个小型音乐播放器。它成本低、功耗低&#xff0c;可…

类与对象(二)

封装 封装作为面向对象三大特性&#xff08;封装&#xff0c;继承&#xff0c;多态&#xff09;之一&#xff0c;那如何实现封装性的呢&#xff1f;就又得拿出上面的访问修饰限定符的图 public: 就是在任何地方都可以访问 protected: 涉及子类在介绍继承时详细介绍 default: …

【深度学习目标检测】二十六、基于深度学习的垃圾检测系统-含数据集、GUI和源码(python,yolov8)

设计垃圾检测系统的意义在于多个方面&#xff0c;这些方面不仅关乎环境保护和城市管理&#xff0c;还涉及到技术进步和社会效益。以下是设计垃圾检测系统的主要意义&#xff1a; 环境保护与资源回收&#xff1a; 垃圾检测系统能够有效地识别不同种类的垃圾&#xff0c;帮助人们…

数据可视化(十二):Pandas太阳黑子数据、图像处理——离散极值、核密度、拟合曲线、奇异值分解等高级操作

Tips&#xff1a;"分享是快乐的源泉&#x1f4a7;&#xff0c;在我的博客里&#xff0c;不仅有知识的海洋&#x1f30a;&#xff0c;还有满满的正能量加持&#x1f4aa;&#xff0c;快来和我一起分享这份快乐吧&#x1f60a;&#xff01; 喜欢我的博客的话&#xff0c;记得…

【JS红宝书学习笔记】第1、2章 初识JS

第1章 什么是JavaScript JavaScript 是一门用来与网页交互的脚本语言&#xff0c;包含以下三个组成部分。 ECMAScript&#xff1a;由 ECMA-262 定义并提供核心功能。文档对象模型&#xff08;DOM&#xff09;&#xff1a;提供与网页内容交互的方法和接口。浏览器对象模型&…

LeetCode 98. 验证二叉搜索树

LeetCode 98. 验证二叉搜索树 1、题目 题目链接&#xff1a;98. 验证二叉搜索树 给你一个二叉树的根节点 root &#xff0c;判断其是否是一个有效的二叉搜索树。 有效 二叉搜索树定义如下&#xff1a; 节点的左子树只包含 小于 当前节点的数。节点的右子树只包含 大于 当前节…

使用apache和htaccess对目录访问设置密码保护配置教程

对目录设置密码保护配置说明 我们有时候访问某些网站的时候&#xff0c;要求输入用户名和密码才能访问。这是为了保护隐私&#xff0c;只让经过许可的人访问。 在本教程中主要介绍两种方法&#xff0c;一种是通过apache httpd.conf配置文件对管理后台目录设置密码保护&#xff…

LeetCode 700.二叉搜索树中的搜索

LeetCode 700.二叉搜索树中的搜索 1、题目 题目链接&#xff1a;700. 二叉搜索树中的搜索 给定二叉搜索树&#xff08;BST&#xff09;的根节点 root 和一个整数值 val。 你需要在 BST 中找到节点值等于 val 的节点。 返回以该节点为根的子树。 如果节点不存在&#xff0c;则…

Docker入门指南:Docker容器的使用(三)

&#x1f340; 前言 博客地址&#xff1a; CSDN&#xff1a;https://blog.csdn.net/powerbiubiu &#x1f44b; 简介 在本章节中&#xff0c;将深入探讨 Docker 容器的概念&#xff0c;以及容器的使用。 &#x1f4d6; 正文 1 什么是容器 1.1 Docker容器的介绍 Docker 容…

使用Gin编写Web API项目并自动化文档

最近需要使用Go写一个Web API项目&#xff0c;可以使用Beego与Gin来写此类项目&#xff0c;前文使用Beego创建API项目并自动化文档介绍了使用Beego来创建的Web API项目并自动化文档的方法。本文就介绍一下使用Gin来编写Web API项目并自动化文档。 一、创建项目 在创建Beego项…

栈与队列OJ题【括号适配问题】【用队列实现栈】【用栈实现队列】【设计循环队列】

一.有效的括号 ​​​OJ链接 这一道题我们就可以用栈来解决&#xff1a; 不了解栈的可以看我的上一篇博客。 typedef char STDataType; //用数组来实现栈 typedef struct stack {STDataType* a;int capacity;int top; }ST; void STInit(ST* pst) {assert(pst);pst->a NU…