博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
存取像素值操作
阅读量:4225 次
发布时间:2019-05-26

本文共 2049 字,大约阅读时间需要 6 分钟。

在上一讲里,对Mat内数据的各种读写操作进行了速度的比较,都是我自己想到的方法,感觉不够系统,这次整理了下思路,参考了文献,把能想到的方法进行了汇总,希望能对大家有所帮助。

1.存取单个像素值

最通常的方法就是

[cpp] 
  1. img.at<uchar>(i,j) = 255;  
  2. img.at<Vec3b>(i,j)[0] = 255;  

如果你觉得at操作显得太笨重了,不想用Mat这个类,也可以考虑使用轻量级的Mat_类,使用重载操作符()实现取元素的操作。

[cpp] 
  1. cv::Mat_<uchar> im2= img; // im2 refers to image  
  2.    im2(50,100)= 0; // access to row 50 and column 100  

2.用指针扫描一幅图像

对于一幅图像的扫描,用at就显得不太好了,还是是用指针的操作方法更加推荐。先介绍一种上一讲提到过的

[cpp] 
  1. for (int j=0; j<nl; j++)  
  2. {  
  3.         uchar* data= image.ptr<uchar>(j);  
  4.         for (int i=0; i<nc; i++)  
  5.        {                   
  6.                   data[i] = 255;  
  7.         }  
  8. }  

更高效的扫描连续图像的做法可能是把W*H的衣服图像看成是一个1*(w*h)的一个一维数组,这个想法是不是有点奇葩,这里要利用
isContinuous
这个函数判断图像内的像素是否填充满,使用方法如下:

[cpp] 
  1. if (img.isContinuous())  
  2. {  
  3.         nc = img.rows*img.cols*img.channels();  
  4. }  
  5. uchar* data = img.ptr<uchar>(0);  
  6. for (int i=0; i<nc; i++)  
  7. {  
  8.         data[i] = 255;  
  9. }  

更低级的指针操作就是使用Mat里的data指针,之前我称之为暴力青年,使用方法如下:

[cpp] 
  1. uchar* data = img.data;  
  2. // img.at(i, j)  
  3. data = img.data + i * img.step + j * img.elemSize();  

3.用迭代器iterator扫描图像

和C++STL里的迭代器类似,Mat的迭代器与之是兼容的。是MatIterator_。声明方法如下:

[cpp] 
  1. cv::MatIterator_<Vec3b> it;  

或者是:

[cpp] 
  1. cv::Mat_<Vec3b>::iterator it;  

扫描图像的方法如下:

[cpp] 
  1. Mat_<Vec3b>::iterator it = img.begin<Vec3b>();  
  2. Mat_<Vec3b>::iterator itend = img.end<Vec3b>();  
  3. for (; it!=itend; it++)  
  4. {  
  5.          (*it)[0] = 255;  
  6. }  

4.高效的scan image方案总结

还是用我们之前使用过的getTickCount、getTickFrequency函数测试速度。这里我就不一一列举我测试的结果了,直接上结论。测试发现,好的编写风格可以提高50%的速度!要想减少程序运行的时间,必要的优化包括如下几个方面:

(1)内存分配是个耗时的工作,优化之

(2)在循环中重复计算已经得到的值,是个费时的工作,优化之;举例:

[cpp] 
  1. int nc = img.cols * img.channels();  
  2. for (int i=0; i<nc; i++)  
  3. {.......}  
  4. //**************************  
  5. for (int i=0; i<img.cols * img.channels(); i++)  
  6. {......}  

后者的速度比前者要慢上好多。

(3)使用迭代器也会是速度变慢,但迭代器的使用可以减少程序错误的发生几率,考虑这个因素,可以酌情优化

(4)at操作要比指针的操作慢很多,所以对于不连续数据或者单个点处理,可以考虑at操作,对于连续的大量数据,不要使用它

(5)扫描连续图像的做法可能是把W*H的衣服图像看成是一个1*(w*h)的一个一维数组这种办法也可以提高速度。短的循环比长循环更高效,即使他们的操作数是相同的

以上的这些优化可能对于大家的程序运行速度提高并不明显,但它们毕竟是个得到速度提升的好的编程策略,希望大家能多采纳。

还有就是利用多线程也可以高效提高运行速度。OpenMP和TBB是两种流行的APT,不过对于多线程的东西,我是有些迷糊的,呵呵

5.整行整列像素值的赋值

对于整行或者整列的数据,可以考虑这种方式处理

[cpp] 
  1. img.row(i).setTo(Scalar(255));  
  2. img.col(j).setTo(Scalar(255));  

http://blog.csdn.net/yang_xian521/article/details/7182185

你可能感兴趣的文章
怎样判断漂亮女孩是不是单身的?
查看>>
独家 | 教你使用Keras on Google Colab(免费GPU)微调深度神经网络
查看>>
独家 | 如何利用大规模无监督数据建立高水平特征?
查看>>
福利 | NVIDIA英伟达免费直播课:带你选择、搭建AI服务器!
查看>>
报名 | 清华园里的AI公开课!第三讲之神经网络
查看>>
教你在Python中构建物体检测系统(附代码、学习资料)
查看>>
报名 | Sven Travis教授设计与人工智能思享会
查看>>
报名 | 瓜子二手车周洲:如何玩转AI赋能下的新零售?
查看>>
经典论文复现 | LSGAN:最小二乘生成对抗网络
查看>>
报名 | 第二届网上行为社会网分析学术研讨会
查看>>
独家 | 用Python Featuretools库实现自动化特征工程(附链接)
查看>>
报名 | 二值化网络 & 基于素描图的图像检索论坛
查看>>
干货 | 清华大学郑方:语音技术用于身份认证的理论与实践
查看>>
近期活动盘点:2018数据与媒介发展论坛、大数据应用中日交流论坛(11.04-11.15)...
查看>>
干货 | 476个PyTorch资源大合集推荐,GitHub超过3600星
查看>>
清华本科生最高荣誉:图说十年特奖得主(附历年名单)
查看>>
报名 | data Artisans、腾讯、阿里、滴滴、美团专家带你认识Flink!
查看>>
Yoshua:深度学习AI迈向人类水平的挑战(附PPT下载)
查看>>
阿里、京东、去哪儿等大厂技术专家齐聚北京,Flink China Meetup再度归来!
查看>>
独家 | 一文盘点数据集市和数据仓库的差异(附链接)
查看>>