【从零学习openCV】opecv操作像素
1. 存取像素值
//在一张图像上增加椒盐噪声,image为输入图像。n为噪点个数
void salt(Mat &image, int n)
{
for(int k = 0;k < n;k++)
{
//随机产生白色噪点
int i = qrand()%image.cols;
int j = qrand()%image.rows;
//假设是灰度图每一个像素的存取类型为uchar,即8bit整型数
if(image.channels() == 1){
image.at<uchar>(j,i) = 255;
}
//彩色图像有三个通道,像素存取类型为cv::Vec3b,即由三个uchar组成的向量。这里用下标[i]訪问每一个通道
else{
image.at<Vec3b>(j,i)[0] = 255;
image.at<Vec3b>(j,i)[1] = 255;
image.at<Vec3b>(j,i)[2] = 255;
}
}
}
效果例如以下:
//颜色缩减函数,image为输入图像。div为缩减的倍数
void colorReduce(Mat&image, int div = 64)
{
int nl = image.rows; //图像的行数
//图像每行的像素数
int nc = image.cols * image.channels();
for(int j =0;j<nl;j++)
{
//得到第j行的首地址
uchar* data = image.ptr<uchar>(j);
//遍历每行的像素
for(int i =0;i<nc;i++)
{
data[i] = data[i]/div*div; //将每一个像素值都变为div的倍数,即将颜色数缩减了div倍
}
}
}
效果例如以下:
//颜色缩减函数。image为输入图像,div为缩减的倍数
void colorReduce(Mat&image, int div = 64)
{
int nl = image.rows; //图像的行数
//图像每行的像素数
int nc = image.cols * image.channels(); //假设图像连续
if(image.isContinuous())
{
//reshape函数用于改变矩阵维度
//图像行数为1,列数为原先的行数乘上列数
image.reshape(1,image.cols*image.rows);
} for(int j =0;j<nl;j++)
{
//得到第j行的首地址
uchar* data = image.ptr<uchar>(j);
//遍历每行的像素
for(int i =0;i<nc;i++)
{
data[i] = data[i]/div*div; //将每一个像素值都变为div的倍数,即将颜色数缩减了div倍
}
}
}
当然opencv还有更为底层的指针。在cv::Mat中。图像数据以unsigned char形式保存在一块内存中。
这块内存的首地址能够通过data成员变量得到。data是一个unsigned char型的指针,所以循环能够用下面方式:
//获得图像指针
uchar *data = image.data;
//获得第j行,第i列个像素值,step代表图像的行宽(包含填补像素)
data = image.data + j*image.step + i*image.elemSize();
//颜色缩减函数。image为输入图像。div为缩减的倍数
void colorReduce(Mat&image, int div = 64)
{
//得到初始位置的迭代器
Mat_<Vec3b>::iterator it = image.begin<Vec3b>();
//得到终止位置的迭代器
Mat_<Vec3b>::iterator itend = image.end<Vec3b>();
//遍历全部像素
for(; it != itend; ++it){
(*it)[0] = (*it)[0]/div*div;
(*it)[1] = (*it)[1]/div*div;
(*it)[2] = (*it)[2]/div*div;
}
}
效果与用指针遍历的一样。
//at方法
void colorReduce1(Mat&image, int div = 64)
{
int nl = image.rows; //图像的行数
//图像每行的像素数
int nc = image.cols * image.channels();
for(int j =0;j<nl-2;j++)
{
for(int i =0;i<nc-2;i++)
{
image.at<Vec3b>(j,i)[0] = image.at<Vec3b>(j,i)[0]/div*div;
image.at<Vec3b>(j,i)[1] = image.at<Vec3b>(j,i)[1]/div*div;
image.at<Vec3b>(j,i)[2] = image.at<Vec3b>(j,i)[2]/div*div;
}
}
} //行首指针方法
void colorReduce2(Mat&image, int div = 64)
{
int nl = image.rows; //图像的行数
//图像每行的像素数
int nc = image.cols * image.channels();
for(int j =0;j<nl;j++)
{
//得到第j行的首地址
uchar* data = image.ptr<uchar>(j);
//遍历每行的像素
for(int i =0;i<nc;i++)
{
data[i] = data[i]/div*div; //将每一个像素值都变为div的倍数,即将颜色数缩减了div倍
}
}
} //一维数组
void colorReduce3(Mat&image, int div = 64)
{
int nl = image.rows; //图像的行数
//图像每行的像素数
int nc = image.cols * image.channels(); //假设图像连续
if(image.isContinuous())
{
//reshape函数用于改变矩阵维度
//图像行数为1,列数为原先的行数乘上列数
image.reshape(1,image.cols*image.rows);
} for(int j =0;j<nl;j++)
{
//得到第j行的首地址
uchar* data = image.ptr<uchar>(j);
//遍历每行的像素
for(int i =0;i<nc;i++)
{
data[i] = data[i]/div*div; //将每一个像素值都变为div的倍数,即将颜色数缩减了div倍
}
}
} //迭代器方法
void colorReduce4(Mat&image, int div = 64)
{
//得到初始位置的迭代器
Mat_<Vec3b>::iterator it = image.begin<Vec3b>();
//得到终止位置的迭代器
Mat_<Vec3b>::iterator itend = image.end<Vec3b>();
//遍历全部像素
for(; it != itend; ++it){
(*it)[0] = (*it)[0]/div*div;
(*it)[1] = (*it)[1]/div*div;
(*it)[2] = (*it)[2]/div*div;
}
} //測试4种像素遍历方式执行时间
void calrunTime(int v,Mat&image)
{
double duration;
duration = static_cast<double>(getTickCount());
for(int i = 0;i<10;i++) //执行十次取平均值
{
switch(v)
{
case 1:
colorReduce1(image);
break;
case 2:
colorReduce2(image);
break;
case 3:
colorReduce3(image);
break;
case 4:
colorReduce4(image);
break;
default:
break;
}
}
duration = static_cast<double>(getTickCount()) - duration;
duration /= getTickFrequency()/100; //执行时间,以ms为单位
qDebug()<<"duration"<<v<<":"<<duration<<"ms";
}
【从零学习openCV】opecv操作像素的更多相关文章
- 【从零学习openCV】IOS7下的人脸检測
前言: 人脸检測与识别一直是计算机视觉领域一大热门研究方向,并且也从安全监控等工业级的应用扩展到了手机移动端的app,总之随着人脸识别技术获得突破,其应用前景和市场价值都是不可估量的,眼下在学习ope ...
- 【从零学习openCV】IOS7根据人脸检测
前言: 人脸检測与识别一直是计算机视觉领域一大热门研究方向,并且也从安全监控等工业级的应用扩展到了手机移动端的app.总之随着人脸识别技术获得突破,其应用前景和市场价值都是不可估量的,眼下在学习ope ...
- 【从零学习openCV】IOS7人脸识别实战
前言 接着上篇<IOS7下的人脸检測>,我们顺藤摸瓜的学习怎样在IOS7下用openCV的进行人脸识别,实际上非常easy,因为人脸检測部分已经完毕,剩下的无非调用openCV的方法对採集 ...
- 【从零学习openCV】IOS7下的openCV开发起步(Xcode5.1.1&openCV2.49)
前言: 开发IOS7已经有一月的时间了.近期在准备推研的事,有点想往CV方向发展.于是開始自学openCV. 关注CSDN已经非常久了.也从非常多博主那学到了非常多知识,于是我也从这周开启自己的blo ...
- 图像混合学习。运用加权函数,学习opencv基础操作
{ cout<< } { cout<< } ,,logoImage.c ...
- 学习Opencv 2.4.9(二) ---操作像素
作者:咕唧咕唧liukun321 来自:http://blog.csdn.net/liukun321 本质上说一张图像就是由数值组成的矩阵.Opencv 2.x由 cv::Mat 这个数据结构来表示一 ...
- OpenCV学习C++接口 Mat像素遍历详解
OpenCV学习C++接口 Mat像素遍历详解
- OpenCV坐标系与操作像素的四种方法
像素是图像的基本组成单位,熟悉了如何操作像素,就能更好的理解对图像的各种处理变换的实现方式了. 1.at方法 第一种操作像素的方法是使用"at",如一幅3通道的彩色图像image的 ...
- 【学习opencv第六篇】图像的反转操作
考试终于完了,现在终于有时间可以继续学习这个了.写这篇博客主要是因为以前一直搞不清楚图像数据到底是怎么存储的,以及这个step到底是什么,后来查了一下才知道原来step就是数据行的长度.. #incl ...
随机推荐
- c++构造析构顺序
class A { public: A(){ cout << "constrcut A" << endl; }; ~A(){ cout << & ...
- 编译安装MongoDB C++ Driver (win8.1 vs2013)
在C++中调用mongodb的库函数需要安装mongodb的c++driver,需要自己编译,(自己搞了一天半 =_=''' ) 官网Build MongoDB From Source 说To bui ...
- 读书笔记:php_tizag_tutorial
昨天在实验室花了一天时间看了英文版的php_tizag_tutorial,因为上学期用php和bootstrap写过一个租房网站,对php还是比较熟悉.现在总结一下php_tizag_tutorial ...
- CVPapers论文整理工具-开源
一.工具介绍及运行实例 相信计算机视觉领域的同道中人都知道这个Computer Vision Resource网站, http://www.cvpapers.com/ 网页部分截图如下: 可以看到有 ...
- 开源OCR光学字符识别
纸张在 许多地方已日益失宠,无纸化办公谈论40多年,办公环境正限制纸山的生成.而过去几年,无纸化办公的概念发生了显着的转变.在计算机软件的帮助 下,包含大量重要管理数据和资讯的文档可以更方便的以电子形 ...
- [置顶] CF 86D Powerful array 分块算法入门,n*sqrt(n)
简介:分块算法主要是把区间划分成sqrt(n)块,从而降低暴力的复杂度, 其实这算是一种优化的暴力吧,复杂度O(n*sqrt(n)) 题意:给定一个数列:a[i] (1<= i <= ...
- 基于visual Studio2013解决面试题之1003字符串逆序
题目
- 基于visual Studio2013解决面试题之0908最大连续数字串
题目
- 怎样在Ubuntu中使用条件布局
我们知道现代手机能够随着手持的方位发生改变而使得手机的方位也随着发生改变.对有些应用来说,我们也希望手机的布局也能尾随发生变化.第二种情况是当我们的应用安装到不同屏幕尺寸的平台上,我们希望我们的布局会 ...
- 一起C语言中程序时序问题的排查过程
[文章摘要] 对于由多个模块协同工作的软件来说,程序处理的时序是很重要的.当消息处理的顺序出现混乱时,程序就会出现异常. 本文基于作者的实际项目经验.对软件模块之间的时序问题进行了具体的分析,为相关软 ...