opencv学习笔记(三):几种去噪滤波器的实现
来源:互联网 发布:mac版su破解教程 编辑:程序博客网 时间:2024/05/31 19:56
现在在上数字图像处理的课程,最近的一次作业要求不用OpenCV自带的滤波器函数来实现几种滤波器,以实现对加入椒盐噪声的图像的去噪。也是对markdown编辑器的一次练习。
椒盐噪声
椒盐噪声是一种很简单的噪声,即随机将图像中一定数量的像素点设置为0(黑)或255(白)。由于看起来好像在图像上撒了椒盐一样,故被称为椒盐噪声。
下面是椒盐噪声的处理代码(假定输入图像为3通道)
void salt(Mat &image, float salt_ratio){ int n=image.rows*image.cols*salt_ratio; for (int k = 0; k < n; k++) { int i = rand() % image.cols; //cols 和 rows 给出图像的宽与高 int j = rand() % image.rows; int type= rand() %2; if (type==1){ image.at<Vec3b>(j, i)[0] = 255; image.at<Vec3b>(j, i)[1] = 255; image.at<Vec3b>(j, i)[2] = 255; } else{ image.at<Vec3b>(j, i)[0] = 0; image.at<Vec3b>(j, i)[1] = 0; image.at<Vec3b>(j, i)[2] = 0; } }}
以经典的lena图为例,加入10%的椒盐噪声后:
滤波器原理
首先介绍一些通用的设定:令
算术均值滤波器
几何均值滤波器
谐波滤波器
中值滤波器
用相邻区域像素灰度的中值代替该点的像素值
编程实现
根据以上公式,实现了四种模板大小可变的滤波器。
主要编程思想为:构建一个ImageRecovery类,其中filter函数为公用的滤波器接口,在filter函数中对result(Mat格式3通道图片)进行遍历,然后根据选择的滤波器类型调用不同的滤波器模板,进行滤波运算。滤波器模板函数则作为类的private函数。
此外,在谐波滤波和几何滤波中,若模板中有一个像素的取值为0,则不将这个点引入计算。若引入计算则会导致错误或是0像素点相邻的一片区域完全变黑。相当于增加了图像的噪声。代码如下:
#include "stdafx.h"#include <opencv2/core/core.hpp>#include <opencv2/highgui/highgui.hpp>#include <opencv2/opencv.hpp>using namespace cv;using namespace std;class ImageRecovery{private: double filter_aver(Mat src) { //算术均值滤波 double sum=0; for (int i =0;i < src.rows;i++ ){ uchar* data=src.ptr<uchar>(i); for (int j =0;j <src.cols;j++){ sum+=double(data[j]); } } return sum/double(src.cols* src.rows); } double filter_geo(Mat src) { //几何均值滤波 double geo=1; for (int i =0;i < src.rows;i++ ){ uchar* data=src.ptr<uchar>(i); for (int j =0;j <src.cols;j++){ if (data[j]!=0) geo*=data[j]; } } double power=1.0/double(src.cols*src.rows); return pow(geo,power); } double filter_har(Mat src) { //谐波滤波 double har=0; for (int i =0;i < src.rows;i++ ){ uchar* data=src.ptr<uchar>(i); for (int j =0;j <src.cols;j++){ if (data[j]!=0) har+=1/(double)(data[j]); } } return (src.cols*src.rows)/har; } void BubbleSort(float* pData, int count) { //冒泡排序,用于中值滤波 float tData; for (int i = 1; i < count; i++){ for (int j = count - 1; j >= i; j--){ if (pData[j] < pData[j - 1]){ tData = pData[j - 1]; pData[j - 1] = pData[j]; pData[j] = tData; } } } } double filter_median(Mat src) { //中值滤波 int index=0; int bubble_len=(src.cols)*(src.rows); float* bubble=new float[bubble_len]; for (int i =0;i < src.rows;i++ ){ uchar* data=src.ptr<uchar>(i); for (int j =0;j <src.cols;j++){ bubble[index] = data[j]; index ++; } } BubbleSort(bubble,bubble_len); double median=bubble[bubble_len/2]; return median; }public: void salt(Mat &image, float salt_ratio ) { //salt_ratio为加入椒盐噪声的比例 int n=image.rows*image.cols*salt_ratio; for (int k = 0; k < n; k++) { int i = rand() % image.cols; int j = rand() % image.rows; int type= rand() %2; if (type==1){ image.at<Vec3b>(j, i)[0] = 255; image.at<Vec3b>(j, i)[1] = 255; image.at<Vec3b>(j, i)[2] = 255; } else{ image.at<Vec3b>(j, i)[0] = 0; image.at<Vec3b>(j, i)[1] = 0; image.at<Vec3b>(j, i)[2] = 0; } } } Mat filter(Mat image,string filter_type,Size size) { //image为输入待滤波图像,filter_tpye为滤波器类型,size为滤波器的尺寸 Mat result; image.copyTo(result); Mat channel[3]; split(image,channel); int l =(size.height-1)/2; int w =(size.width-1)/2; for (int i = l;i < result.rows-l;i ++){ for (int j =w;j < result.cols-w;j ++){ for (int ii =0;ii < 3;ii++){ if (filter_type=="aver") result.at<Vec3b>(i,j)[ii]=saturate_cast<uchar>(filter_aver(channel[ii](Rect(j-w,i-l,size.width,size.height)))); if (filter_type=="geo" ) result.at<Vec3b>(i,j)[ii]=saturate_cast<uchar>(filter_geo(channel[ii](Rect(j-w,i-l,size.width,size.height)))); if (filter_type=="har" ) result.at<Vec3b>(i,j)[ii]=saturate_cast<uchar>(filter_har(channel[ii](Rect(j-w,i-l,size.width,size.height)))); if (filter_type=="median" ) result.at<Vec3b>(i,j)[ii]=saturate_cast<uchar>(filter_median(channel[ii](Rect(j-w,i-l,size.width,size.height)))); } } } return result; }};int main(){ Mat img=imread("lena.jpg"); //初始化IR类 ImageRecovery IR; //加入椒盐噪声 IR.salt(img,0.1); imshow("salt",img); //对噪声图片进行滤波 Mat result=IR.filter(img,"geo",Size(3,3)); imshow("result",result); waitKey(); return 0;}
效果如图所示
算术均值滤波(左:3*3;右:5*5)
几何均值滤波(左:3*3;右:5*5)
谐波滤波(左:3*3;右:5*5)
中值滤波(左:3*3;右:5*5)
总的来看,基于个人主观的判断,我认为对于椒盐噪声图片的恢复,中值滤波的效果最好,谐波滤波的效果次之,均值滤波的效果再次,几何滤波的效果最差。
而对于滤波器的不同模板大小,从结果上可以看出选择更大的模板可以使得结果图像中的噪声更加平滑,但同时图像也变得模糊了。这是由于更大的滤波器模板使得图像整体变得平滑了。
这次也是我第一次使用markdown写博客,线下使用wiznote的markdown模式进行编辑,图片则存储在七牛云上。这样在线下写完的博客可以直接复制到csdn上,省去了我之前每次上传博客还要改格式以及传图片的烦恼。此外,用markdown写感觉格式上也好看了很多,特别是公式方面。
- opencv学习笔记(三):几种去噪滤波器的实现
- opencv学习笔记(5)----滤波器
- Opencv学习笔记(一)滤波器原理及其使用
- 我的OpenCV学习笔记(10):空域滤波之锐化滤波器
- OpenCV 2.2 的学习笔记(三)
- OpenCV中滤波器学习笔记(一)--CvSepFilter
- OpenCV学习----深入理解双边滤波器实现
- OpenCV 学习笔记(三)
- opencv学习笔记(三)
- opencv学习笔记(三)
- OpenCV学习笔记(三)
- 学习笔记--滤波器的系数
- opencv学习-imgprocess-创建自己的滤波器
- gabor滤波器 opencv 实现
- opencv实现中值滤波器
- opencv学习笔记三
- opencv学习笔记三
- Opencv Qt 学习笔记(三) 级联器的训练
- UOJ #244. 【UER #7】短路
- 营养与锌
- C语言初学----对圆 y=x^2 以及两者共存图像的输出 对于任意一个数 求出最大公约数
- Android全屏(包含3种隐藏顶部状态栏及标题栏和一种隐藏Android 4.0平板底部状态栏的方法)
- java基础之类加载器
- opencv学习笔记(三):几种去噪滤波器的实现
- 段错误bug的调试
- hibernate4整合spring3出现java.lang.NoClassDefFoundError: [Lorg/hibernate/engine/FilterDefinition;
- setContentView(),inflate()方法
- 单片机
- 关于MVP模式的使用设计
- [AHK]华泰证券的委托认证和用户登录窗口的自动处理
- 说说心事(一)
- OAuth2.0 Server 设计