opencv函数GaussianBlur,pyrDown,pyrUp深究

2023-10-21 21:38

本文主要是介绍opencv函数GaussianBlur,pyrDown,pyrUp深究,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

通常在使用GaussianBlur,pyrDown,pyrUp的时候都不会深究的内部实现,最近想自己实现一下,看看会遇到哪些问题,然后将结果和opencv的函数运行结果进行对比。
C++: void GaussianBlur(InputArray src, OutputArray dst, Size ksize, double sigmaX, double sigmaY=0, int borderType=BORDER_DEFAULT )

从GaussianBlur的参数可以看出,有sigmaX 和sigmaY, 二维高斯核使用的是分别在X和Y方向进行一维高斯核卷积来模拟二维高斯核的效果。

对于边界问题,BORDER_DEFAULT 是指:pixel[-1]使用pixel[1]的像素值来代替,pixel[-2] = pixel[2],pixel[width] = pixel[width - 2],依此类推(如果pixel[0],pixel[width -1]为边界的话) 。

	for (int row = 0 ;row <src.rows ;row++)//x orientation{for (int col = 0 ; col <src.cols; col++){float sumRed = 0,sumGreen = 0,sumBlue = 0;for (int subcol = -win_size;subcol<=win_size;subcol++){int srccol = col+subcol;if (srccol<0){srccol *= -1;}if (srccol>=src.cols){srccol = src.cols*2 - srccol - 2;}sumRed += kernel[subcol+win_size]*src.at<Vec3b>(row,srccol)[2];sumGreen += kernel[subcol+win_size]*src.at<Vec3b>(row,srccol)[1];sumBlue += kernel[subcol+win_size]*src.at<Vec3b>(row,srccol)[0];}temp.at<Vec3b>(row,col)[2] = sumRed;temp.at<Vec3b>(row,col)[1] = sumGreen;temp.at<Vec3b>(row,col)[0] = sumBlue;}}for (int row = 0 ;row <src.rows ;row++)//y orientation{for (int col = 0 ; col <src.cols; col++){float sumRed = 0,sumGreen = 0,sumBlue = 0;for (int subrow = -win_size;subrow<=win_size;subrow++){int srcrow = row+subrow;if (srcrow<0){srcrow *= -1;}if (srcrow>=src.rows){srcrow = src.rows*2 - srcrow - 2;;}sumRed += kernel[subrow+win_size]*temp.at<Vec3b>(srcrow,col)[2];sumGreen += kernel[subrow+win_size]*temp.at<Vec3b>(srcrow,col)[1];sumBlue += kernel[subrow+win_size]*temp.at<Vec3b>(srcrow,col)[0];}result.at<Vec3b>(row,col)[2] = sumRed;result.at<Vec3b>(row,col)[1] = sumGreen;result.at<Vec3b>(row,col)[0] = sumBlue;}}

C++: void pyrDown(InputArray src, OutputArray dst, const Size& dstsize=Size())

pyrDown 是降采样,分两步进行。

第一步:对src图像进行模糊,使用的模糊核不是高斯核,而是这样一个核函数


第二步:模糊之后的图像去除所有偶数行和偶数列。
void GassianPyramid::downSampleFunc(Mat src, Mat& result)
{int width = src.cols;int height = src.rows;int ow = width/2;  int oh = height/2;  result.create(Size(ow,oh),src.type());float tempArr[5][5]={1,4,6,4,1,4,16,24,16,4,6,24,36,24,6,4,16,24,16,4,1,4,6,4,1};for (int i = 0;i<5;i++){for (int j = 0; j<5; j++){tempArr[i][j] /=256;}}Mat temp;temp.create(Size(src.cols,src.rows),src.type());for (int row = 0 ;row <src.rows ;row++){for (int col = 0 ; col <src.cols; col++){float sumRed = 0,sumGreen = 0,sumBlue = 0;for (int subrow = -win_size;subrow<=win_size;subrow++){int srcrow = row+subrow;if (srcrow<0){srcrow *= -1;}if (srcrow>=src.rows){srcrow = (src.rows - 1) - (srcrow - (src.rows-1));}for (int subcol = -win_size;subcol<=win_size;subcol++){int srccol = col+subcol;if (srccol<0){srccol *= -1;}if (srccol>=src.cols){srccol = (src.cols - 1) - (srccol - (src.cols-1));}sumRed += tempArr[subrow+win_size][subcol+win_size]*src.at<Vec3b>(srcrow,srccol)[2];sumGreen += tempArr[subrow+win_size][subcol+win_size]*src.at<Vec3b>(srcrow,srccol)[1];sumBlue += tempArr[subrow+win_size][subcol+win_size]*src.at<Vec3b>(srcrow,srccol)[0];}}temp.at<Vec3b>(row,col)[2] = sumRed;temp.at<Vec3b>(row,col)[1] = sumGreen;temp.at<Vec3b>(row,col)[0] = sumBlue;}}int inRow=0, inCol = 0;  		for(int row=0; row<oh; row++) {  for(int col=0; col<ow; col++) {  inRow = 2* row;  inCol = 2* col;  if(inRow >= height) {  inRow = height;  }  if(inCol >= width) {  inCol = width;  }  result.at<Vec3b>(row,col)[2]= temp.at<Vec3b>(inRow,inCol)[2];result.at<Vec3b>(row,col)[1]= temp.at<Vec3b>(inRow,inCol)[1];result.at<Vec3b>(row,col)[0]= temp.at<Vec3b>(inRow,inCol)[0];}  }
}

C++: void pyrUp(InputArray src, OutputArray dst, const Size& dstsize=Size())
pyrUp是上采样,也是分两步进行。第一步:将图像在每个方向扩大为原来的两倍,新增的行和列以0填充。

 第二步:将 
上面降采样的高斯核*4和放大后的图像进行卷积。 
void GassianPyramid::expendSampleFunc(Mat src, Mat& result)
{int width = src.cols;int height = src.rows;int ow = 2*width;  int oh =2*height; Mat temp;temp.create(Size(ow,oh),src.type());result.create(Size(ow,oh),src.type());for(int row=0; row<oh; row++) {  for(int col=0; col<ow; col++) {  int inRow = row/2;  int inCol = col/2;if(row - inRow*2 > 0 || col - inCol*2 > 0) {  temp.at<Vec3b>(row,col)[2] = 0;temp.at<Vec3b>(row,col)[1] = 0;temp.at<Vec3b>(row,col)[0] = 0;continue;} temp.at<Vec3b>(row,col)[2] = src.at<Vec3b>(inRow,inCol)[2];temp.at<Vec3b>(row,col)[1] = src.at<Vec3b>(inRow,inCol)[1];temp.at<Vec3b>(row,col)[0] = src.at<Vec3b>(inRow,inCol)[0];}  }  float tempArr[5][5]={1,4,6,4,1,4,16,24,16,4,6,24,36,24,6,4,16,24,16,4,1,4,6,4,1};for (int i = 0;i<5;i++){for (int j = 0; j<5; j++){tempArr[i][j] /=64;}}for (int row = 0 ;row <temp.rows ;row++){for (int col = 0 ; col <temp.cols; col++){float sumRed = 0,sumGreen = 0,sumBlue = 0;for (int subrow = -win_size;subrow<=win_size;subrow++){int srcrow = row+subrow;if (srcrow<0){srcrow *= -1;}if (srcrow>=temp.rows){srcrow = (temp.rows - 1) - (srcrow - (temp.rows-1));}for (int subcol = -win_size;subcol<=win_size;subcol++){int srccol = col+subcol;if (srccol<0){srccol *= -1;}if (srccol>=temp.cols){srccol = (temp.cols - 1) - (srccol - (temp.cols-1));}sumRed += tempArr[subrow+win_size][subcol+win_size]*temp.at<Vec3b>(srcrow,srccol)[2];sumGreen += tempArr[subrow+win_size][subcol+win_size]*temp.at<Vec3b>(srcrow,srccol)[1];sumBlue += tempArr[subrow+win_size][subcol+win_size]*temp.at<Vec3b>(srcrow,srccol)[0];}}result.at<Vec3b>(row,col)[2] = sumRed;result.at<Vec3b>(row,col)[1] = sumGreen;result.at<Vec3b>(row,col)[0] = sumBlue;}}
}


结果发现,和opencv自带的函数运行结果相同,证明结果正确。







这篇关于opencv函数GaussianBlur,pyrDown,pyrUp深究的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

MySQL 日期时间格式化函数 DATE_FORMAT() 的使用示例详解

《MySQL日期时间格式化函数DATE_FORMAT()的使用示例详解》`DATE_FORMAT()`是MySQL中用于格式化日期时间的函数,本文详细介绍了其语法、格式化字符串的含义以及常见日期... 目录一、DATE_FORMAT()语法二、格式化字符串详解三、常见日期时间格式组合四、业务场景五、总结一、

golang panic 函数用法示例详解

《golangpanic函数用法示例详解》在Go语言中,panic用于触发不可恢复的错误,终止函数执行并逐层向上触发defer,最终若未被recover捕获,程序会崩溃,recover用于在def... 目录1. panic 的作用2. 基本用法3. recover 的使用规则4. 错误处理建议5. 常见错

Python itertools中accumulate函数用法及使用运用详细讲解

《Pythonitertools中accumulate函数用法及使用运用详细讲解》:本文主要介绍Python的itertools库中的accumulate函数,该函数可以计算累积和或通过指定函数... 目录1.1前言:1.2定义:1.3衍生用法:1.3Leetcode的实际运用:总结 1.1前言:本文将详

轻松上手MYSQL之JSON函数实现高效数据查询与操作

《轻松上手MYSQL之JSON函数实现高效数据查询与操作》:本文主要介绍轻松上手MYSQL之JSON函数实现高效数据查询与操作的相关资料,MySQL提供了多个JSON函数,用于处理和查询JSON数... 目录一、jsON_EXTRACT 提取指定数据二、JSON_UNQUOTE 取消双引号三、JSON_KE

MySQL数据库函数之JSON_EXTRACT示例代码

《MySQL数据库函数之JSON_EXTRACT示例代码》:本文主要介绍MySQL数据库函数之JSON_EXTRACT的相关资料,JSON_EXTRACT()函数用于从JSON文档中提取值,支持对... 目录前言基本语法路径表达式示例示例 1: 提取简单值示例 2: 提取嵌套值示例 3: 提取数组中的值注意

Java function函数式接口的使用方法与实例

《Javafunction函数式接口的使用方法与实例》:本文主要介绍Javafunction函数式接口的使用方法与实例,函数式接口如一支未完成的诗篇,用Lambda表达式作韵脚,将代码的机械美感... 目录引言-当代码遇见诗性一、函数式接口的生物学解构1.1 函数式接口的基因密码1.2 六大核心接口的形态学

Java中的Opencv简介与开发环境部署方法

《Java中的Opencv简介与开发环境部署方法》OpenCV是一个开源的计算机视觉和图像处理库,提供了丰富的图像处理算法和工具,它支持多种图像处理和计算机视觉算法,可以用于物体识别与跟踪、图像分割与... 目录1.Opencv简介Opencv的应用2.Java使用OpenCV进行图像操作opencv安装j

Oracle的to_date()函数详解

《Oracle的to_date()函数详解》Oracle的to_date()函数用于日期格式转换,需要注意Oracle中不区分大小写的MM和mm格式代码,应使用mi代替分钟,此外,Oracle还支持毫... 目录oracle的to_date()函数一.在使用Oracle的to_date函数来做日期转换二.日

opencv实现像素统计的示例代码

《opencv实现像素统计的示例代码》本文介绍了OpenCV中统计图像像素信息的常用方法和函数,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录1. 统计像素值的基本信息2. 统计像素值的直方图3. 统计像素值的总和4. 统计非零像素的数量

C++11的函数包装器std::function使用示例

《C++11的函数包装器std::function使用示例》C++11引入的std::function是最常用的函数包装器,它可以存储任何可调用对象并提供统一的调用接口,以下是关于函数包装器的详细讲解... 目录一、std::function 的基本用法1. 基本语法二、如何使用 std::function