OpenCV学习:矩阵的掩码操作
来源:互联网 发布:java total memory 编辑:程序博客网 时间:2024/04/30 11:07
Opencv_tutorials学习笔记
矩阵的掩码操作
矩阵掩码操作的思想:根据掩码矩阵(也称作核)重新计算图像中每个像素的值。掩码矩阵中的值表示近邻像素值(包括该像素自身的值)对新像素值有多大影响。从数学观点看,我们用自己设置的权值,对像素邻域内的值做了个加权平均。
测试用例是一个关于图像增强的例子
对每个像素应用下面的公式进行处理
实现掩码操作的两种方法。一种方法是用基本的像素访问方法,另一种方法是用 filter2D 函数。
基本的像素访问方法是利用指针遍历图像,使用[]操作符,就能轻松访问到目标元素。为了让输出指针向前移动,我们在每一次操作之后对输出指针进行了递增(移动一个字节)。
而filter2D函数是opencv自带的函数,必须先定义一个表示掩码的Mat对象
Mat kern = (Mat_<char>(3,3) << 0, -1, 0, -1, 5, -1, 0, -1, 0);
然后调用filter2D函数,参数包括输入、输出图像以及用到的核:
filter2D(I, K, I.depth(), kern );还带有第五个可选参数——指定核的中心,和第六个可选参数——指定函数在未定义区域(边界)的行为。使用该函数有一些优点,如代码更加清晰简洁、通常比 自己实现的方法 速度更快(因为有一些专门针对它实现的优化技术)等等。
附录:opencv提供的函数实现(自己写的注释,请谨慎参考)
#include <opencv2/core/core.hpp>//opencv头文件#include <opencv2/highgui/highgui.hpp>#include <opencv2/imgproc/imgproc.hpp>#include <iostream>//流输入输出头文件using namespace std;//使用命名空间using namespace cv;static void help(char* progName){ cout << endl << "This program shows how to filter images with mask: the write it yourself and the" << "filter2d way. " << endl << "Usage:" << endl << progName << " [image_name -- default lena.jpg] [G -- grayscale] " << endl << endl;}void Sharpen(const Mat& myImage,Mat& Result);//函数说明int main( int argc, char* argv[]){ help(argv[0]);//命令行参数数目大于等于2的话,读入第二个参数argv[1]的图像,否则读入的是后面设定路径的图像 //const char* filename = argc >=2 ? argv[1] : "lena.jpg";const char* filename = argc >=2 ? argv[1] : "D:\\files\\C_exe\\Img_exercise\\lena.jpg"; Mat I, J, K; if (argc >= 3 && !strcmp("G", argv[2]))//参数数目大于等于3的话,比较G和第三个参数argc[2]的值,相等的话imread的是灰度图 I = imread( filename, CV_LOAD_IMAGE_GRAYSCALE);//灰度图设定参数CV_LOAD_IMAGE_GRAYSCALE else I = imread( filename, CV_LOAD_IMAGE_COLOR);//彩色图像设定参数CV_LOAD_IMAGE_COLOR namedWindow("Input", WINDOW_AUTOSIZE); namedWindow("Output1", WINDOW_AUTOSIZE);namedWindow("Output2", WINDOW_AUTOSIZE); imshow("Input", I); double t = (double)getTickCount(); Sharpen(I, J); t = ((double)getTickCount() - t)/getTickFrequency(); cout << "Hand written function times passed in seconds: " << t << endl; imshow("Output1", J); Mat kern = (Mat_<char>(3,3) << 0, -1, 0,//创建一个掩码 -1, 5, -1, 0, -1, 0); t = (double)getTickCount(); filter2D(I, K, I.depth(), kern );//调用filter2D函数 t = ((double)getTickCount() - t)/getTickFrequency(); cout << "Built-in filter2D time passed in seconds: " << t << endl; imshow("Output2", K); waitKey(0); return 0;}void Sharpen(const Mat& myImage,Mat& Result){//使用了 CV_Assert 函数。若该函数括号内的表达式为false,则会抛出一个错误//确保输入图像是无符号字符类型的 CV_Assert(myImage.depth() == CV_8U); // accept only uchar images const int nChannels = myImage.channels();//获取待处理图像的通道数 Result.create(myImage.size(),myImage.type());//用待处理图像的信息create Result图像的矩阵头信息 for(int j = 1 ; j < myImage.rows-1; ++j)//遍历待处理图像除边缘两行外的每一行 { const uchar* previous = myImage.ptr<uchar>(j - 1);//设置三个指针,分别指向相邻的三行首地址 const uchar* current = myImage.ptr<uchar>(j ); const uchar* next = myImage.ptr<uchar>(j + 1); uchar* output = Result.ptr<uchar>(j);//设置指向中间行的指针,即掩模矩阵的中间行所在位置//Q_yyt:为什么不需要对RGB各个通道区分处理 for(int i= nChannels;i < nChannels*(myImage.cols-1); ++i)//i从nChannels开始,略去图像第一列,同样略去图像最后一列 { *output++ = saturate_cast<uchar>(5*current[i]//用 saturate_cast 对结果进行转换,以确保它为有效值 -current[i-nChannels] - current[i+nChannels] - previous[i] - next[i]); } }//公式对边界点来说是未定义的,一种简单的解决方法,是不对这些边界点使用掩码,而直接把它们设为0 Result.row(0).setTo(Scalar(0)); Result.row(Result.rows-1).setTo(Scalar(0)); Result.col(0).setTo(Scalar(0)); Result.col(Result.cols-1).setTo(Scalar(0));}
运行结果:
输入图像
基本方法输出图像
filter2D函数处理输出图像
运行时间比较
0 0
- OpenCV学习笔记:矩阵的掩码操作
- OpenCV学习:矩阵的掩码操作
- OpenCV学习之四: 矩阵的掩码操作
- 7.OpenCv矩阵的掩码操作
- Opencv之矩阵的掩码操作
- Opencv之矩阵的掩码操作
- 学习OpenCV(三)用filter2D实现矩阵掩码操作
- 学习OpenCV(三)用filter2D实现矩阵掩码操作
- 矩阵的掩码操作
- 矩阵的掩码操作
- 矩阵的掩码操作
- 学习OpenCV范例(三)——矩阵的掩码操作
- opencv c++函数 基础3 矩阵的掩码操作
- OpenCV实践(2)- 矩阵的掩码操作
- 理解矩阵的掩码操作 使用opencv锐化图片
- opencv:图像的掩码操作
- core核心模块--矩阵的掩码操作
- opencv(C++)在矩阵上进行掩码操作
- Android异步操作二(AsyncTask)
- cordova文档
- gmail被封了,里面的邮件怎么办?
- 【Manifest】Android获取Manifest中<meta-data>元素的值
- 12-30计划
- OpenCV学习:矩阵的掩码操作
- Android中Parcelable接口和Serial接口的用法与区别
- ie window.print(); 打印总结
- 限制用户输入的用户名密码邮箱格式(正则表达式)
- iOS 字符串常用方法总结——不定时更新
- 搭建本地SVN版本管理工具
- js 实现时间的翻牌效果
- 如何关闭MySQL日志,删除mysql日志
- 一台服务器如何部署多个jboss