转载请注明出处!!!http://blog.csdn.net/zhonghuan1992

OpenCV, colorreduction method

目标:

这次学习的目标是回答以下的几个问题:

1 图片像素是怎样被扫描的?

2OpenCV 矩阵值怎样被存储?

3怎样衡量算法的性能?

4什么是查找表和为什么要用他们?

看完这篇,希望可以解决上面的这些问题。

正文:

首先我们考虑一下简单的色彩减少方法(color reduction method,翻译的不好请指正),假设使用的是c或c++无符号的char(八字节大小的空间),一个信道(channel)有256个不同的值(2^8=256),可是假设使用的是GRB方案,三个channel的话,颜色的数量就会变为256*256*256,大概是16个million这么多,这么多的颜色数量,对于计算机来说仍然是一个负担,所以能够想一些方法来减少这些色彩数量。

能够使用简单的方法来减少图像色彩空间,比方,将0-9的数字都统一用0来取代,10-19的数字都统一用10取代。这样的转换方案能够用以下的公式表示

通过上面的公式,把全部像素点的值更新一下。可是,上面的公式中有除法,这里要表达一个是,计算量比較多的情况下,不用乘除,就不要用,最好把他们转换为加减。我们知道,在转换前像素点的值仅仅有256个,所以我们能够用查找表的方式,我们事先把全部的计算结果都保存在一个数组里,每次要运行上面的公式计算的时候,结果直接从数组里取出来就ok了。比方32相应30,表table[32]=30是早计算出来的,直接訪问table[32]就OK了。

图片矩阵怎样在内存中存储的:

灰度图片的矩阵存储方式:

灰度图片的每个像素点,仅仅由一个值来表示,所以,就是一个普通的二维矩阵。

彩色图片的矩阵存储方式:

彩色图片的存储方式和灰度图片不一样,这里展示的是RGB格式的,能够看到,每个像素,由三个值,代表蓝色,绿色,红色的三个数值表示,存储方式不是三维的,而是二维,只是列向量放大了三倍。从图片中能够清楚的看到。

效率:

比較像素数量减少方式效率的代码,在本文的最后面,代码看上去非常多,事实上结构比較简单,看一会儿就明确了。附上一张结果图:

最快的OpenCV内的LUT函数。关于LUT,看这里

能够粗略的看一下代码,代码不难,非常easy懂:

  1. #include <opencv2/core/core.hpp>
  2. #include <opencv2/highgui/highgui.hpp>
  3. #include <iostream>
  4. #include <sstream>
  5.  
  6. using namespace std;
  7. using namespace cv;
  8.  
  9. static void help()
  10. {
  11. //这里提示输入有三个參数,第一个是图像的名字,第二个是參数是公式中的减少颜色数的数字,这里是10,第三个參数,假设是[G]代表是灰度图片,否则不是。
  12. cout
  13. << "\n--------------------------------------------------------------------------" << endl
  14. << "This program shows how to scan image objects in OpenCV (cv::Mat). As use case"
  15. << " we take an input image and divide the native color palette (255) with the " << endl
  16. << "input. Shows C operator[] method, iterators and at function for on-the-fly item address calculation."<< endl
  17. << "Usage:" << endl
  18. << "./howToScanImages imageNameToUse divideWith [G]" << endl
  19. << "if you add a G parameter the image is processed in gray scale" << endl
  20. << "--------------------------------------------------------------------------" << endl
  21. << endl;
  22. }
  23.  
  24. Mat& ScanImageAndReduceC(Mat& I, const uchar* table);
  25. Mat& ScanImageAndReduceIterator(Mat& I, const uchar* table);
  26. Mat& ScanImageAndReduceRandomAccess(Mat& I, const uchar * table);
  27.  
  28. /*
  29. 程序主要是看不同的color reduction方式对于程序执行速度的影响。
  30. 使用getTickCount()函数来获取当前时间,利用当前时间-上次获取的时间,来得到执行时间
  31.  
  32. */
  33. int main( int argc, char* argv[])
  34. {
  35. help();
  36. if (argc < 3)
  37. {
  38. cout << "Not enough parameters" << endl;
  39. return -1;
  40. }
  41.  
  42. Mat I, J;
  43. if( argc == 4 && !strcmp(argv[3],"G") )
  44. I = imread(argv[1], CV_LOAD_IMAGE_GRAYSCALE);
  45. else
  46. I = imread(argv[1], CV_LOAD_IMAGE_COLOR);
  47.  
  48. if (!I.data)
  49. {
  50. cout << "The image" << argv[1] << " could not be loaded." << endl;
  51. return -1;
  52. }
  53.  
  54. int divideWith = 0; // convert our input string to number - C++ style
  55. stringstream s; //使用stringstream来负责将參数转换为数字
  56. s << argv[2];
  57. s >> divideWith;
  58. if (!s || !divideWith)
  59. {
  60. cout << "Invalid number entered for dividing. " << endl;
  61. return -1;
  62. }
  63.  
  64. uchar table[256];
  65. for (int i = 0; i < 256; ++i)
  66. table[i] = (uchar)(divideWith * (i/divideWith));
  67.  
  68. const int times = 100;
  69. double t;
  70.  
  71. t = (double)getTickCount();
  72.  
  73. for (int i = 0; i < times; ++i)
  74. {
  75. cv::Mat clone_i = I.clone();
  76. J = ScanImageAndReduceC(clone_i, table);
  77. }
  78.  
  79. t = 1000*((double)getTickCount() - t)/getTickFrequency();
  80. t /= times;
  81.  
  82. cout << "Time of reducing with the C operator [] (averaged for "
  83. << times << " runs): " << t << " milliseconds."<< endl;
  84.  
  85. t = (double)getTickCount();
  86.  
  87. for (int i = 0; i < times; ++i)
  88. {
  89. cv::Mat clone_i = I.clone();
  90. J = ScanImageAndReduceIterator(clone_i, table);
  91. }
  92.  
  93. t = 1000*((double)getTickCount() - t)/getTickFrequency();
  94. t /= times;
  95.  
  96. cout << "Time of reducing with the iterator (averaged for "
  97. << times << " runs): " << t << " milliseconds."<< endl;
  98.  
  99. t = (double)getTickCount();
  100.  
  101. for (int i = 0; i < times; ++i)
  102. {
  103. cv::Mat clone_i = I.clone();
  104. ScanImageAndReduceRandomAccess(clone_i, table);
  105. }
  106.  
  107. t = 1000*((double)getTickCount() - t)/getTickFrequency();
  108. t /= times;
  109.  
  110. cout << "Time of reducing with the on-the-fly address generation - at function (averaged for "
  111. << times << " runs): " << t << " milliseconds."<< endl;
  112.  
  113. Mat lookUpTable(1, 256, CV_8U);
  114. uchar* p = lookUpTable.data;
  115. for( int i = 0; i < 256; ++i)
  116. p[i] = table[i];
  117.  
  118. t = (double)getTickCount();
  119.  
  120. for (int i = 0; i < times; ++i)
  121. LUT(I, lookUpTable, J);
  122.  
  123. t = 1000*((double)getTickCount() - t)/getTickFrequency();
  124. t /= times;
  125.  
  126. cout << "Time of reducing with the LUT function (averaged for "
  127. << times << " runs): " << t << " milliseconds."<< endl;
  128. return 0;
  129. }
  130.  
  131. Mat& ScanImageAndReduceC(Mat& I, const uchar* const table)
  132. {
  133. // accept only char type matrices
  134. CV_Assert(I.depth() != sizeof(uchar));
  135.  
  136. int channels = I.channels();
  137.  
  138. int nRows = I.rows;
  139. int nCols = I.cols * channels;
  140.  
  141. if (I.isContinuous())
  142. {
  143. nCols *= nRows;
  144. nRows = 1;
  145. }
  146.  
  147. int i,j;
  148. uchar* p;
  149. for( i = 0; i < nRows; ++i)
  150. {
  151. p = I.ptr<uchar>(i);
  152. for ( j = 0; j < nCols; ++j)
  153. {
  154. p[j] = table[p[j]];
  155. }
  156. }
  157. return I;
  158. }
  159.  
  160. Mat& ScanImageAndReduceIterator(Mat& I, const uchar* const table)
  161. {
  162. // accept only char type matrices
  163. CV_Assert(I.depth() != sizeof(uchar));
  164.  
  165. const int channels = I.channels();
  166. switch(channels)
  167. {
  168. case 1:
  169. {
  170. MatIterator_<uchar> it, end;
  171. for( it = I.begin<uchar>(), end = I.end<uchar>(); it != end; ++it)
  172. *it = table[*it];
  173. break;
  174. }
  175. case 3:
  176. {
  177. MatIterator_<Vec3b> it, end;
  178. for( it = I.begin<Vec3b>(), end = I.end<Vec3b>(); it != end; ++it)
  179. {
  180. (*it)[0] = table[(*it)[0]];
  181. (*it)[1] = table[(*it)[1]];
  182. (*it)[2] = table[(*it)[2]];
  183. }
  184. }
  185. }
  186.  
  187. return I;
  188. }
  189.  
  190. Mat& ScanImageAndReduceRandomAccess(Mat& I, const uchar* const table)
  191. {
  192. // accept only char type matrices
  193. CV_Assert(I.depth() != sizeof(uchar));
  194.  
  195. const int channels = I.channels();
  196. switch(channels)
  197. {
  198. case 1:
  199. {
  200. for( int i = 0; i < I.rows; ++i)
  201. for( int j = 0; j < I.cols; ++j )
  202. I.at<uchar>(i,j) = table[I.at<uchar>(i,j)];
  203. break;
  204. }
  205. case 3:
  206. {
  207. Mat_<Vec3b> _I = I;
  208.  
  209. for( int i = 0; i < I.rows; ++i)
  210. for( int j = 0; j < I.cols; ++j )
  211. {
  212. _I(i,j)[0] = table[_I(i,j)[0]];
  213. _I(i,j)[1] = table[_I(i,j)[1]];
  214. _I(i,j)[2] = table[_I(i,j)[2]];
  215. }
  216. I = _I;
  217. break;
  218. }
  219. }
  220.  
  221. return I;
  222. }

OpenCV, color reduction method的更多相关文章

  1. 35.按要求编写Java程序: (1)编写一个接口:InterfaceA,只含有一个方法int method(int n); (2)编写一个类:ClassA来实现接口InterfaceA,实现int method(int n)接口方 法时,要求计算1到n的和; (3)编写另一个类:ClassB来实现接口InterfaceA,实现int method(int n)接口 方法时,要求计算n的阶乘(n

      35.按要求编写Java程序: (1)编写一个接口:InterfaceA,只含有一个方法int method(int n): (2)编写一个类:ClassA来实现接口InterfaceA,实现in ...

  2. (转)在Mac下使用OpenCV, 在Xcode下使用OpenCV (非常基础,详细)

    转自:http://blog.sciencenet.cn/home.php?COLLCC=3456986939&COLLCC=3456885714&mod=space&uid= ...

  3. 【mac osx安装opencv,python总结】

    在macosx下安装opencv,最大的困难在于协调python版本.由于在opencv官网上,强烈建议安装完整版的python(不建议使用mac 内置的python),所以会碰到这个多个python ...

  4. 学习OpenCV,看这些!

    OpenCV简介: OpenCV 是一款功能强大的跨平台计算机视觉开源库,可以用于解决人机交互.物体检测.人脸识别等领域的问题.库本身是采用 C++ 编写的,但是同时也对 Python, Java, ...

  5. TensorFlow与OpenCV,读取图片,进行简单操作并显示

    TensorFlow与OpenCV,读取图片,进行简单操作并显示 1 OpenCV读入图片,使用tf.Variable初始化为tensor,加载到tensorflow对图片进行转置操作,然后openc ...

  6. VS2015配置OpenCV,使用mfc摄像头程序测试

    转自:https://blog.csdn.net/Lee_Dk/article/details/80466523 这只是介绍了如何加入OpenCV,怎么查找OpenCV请看出处. 新建一个项目.找到属 ...

  7. python base64 编解码,转换成Opencv,PIL.Image图片格式

    二进制打开图片文件,base64编解码,转成Opencv格式: # coding: utf-8 import base64 import numpy as np import cv2 img_file ...

  8. vs2008编译opencv,不能copy CMakeVSMacros2.vsmacros

    由于学习opencv,要查看源码文件,所以要先对opencv进行编译,可悲的是出错了 “不能copy   CMakeVSMacros2.vsmacros” 通过上网查找资料,之所以出现这种情况,是因为 ...

  9. postman发送post请求,报405 Method Not Allowed

    postman发送post请求,报405 Method Not Allowed: 亲测有效的方案一: 检查请求header是否缺少必要信息.如果不清可以把所有的头部信息全部粘贴到header中,尝试是 ...

随机推荐

  1. Android之后台服务判断本应用Activity是否处于栈顶

    在Android开发中,我们经常想知道是否自己的服务处于后台运行中,因为在后台运行的服务器优先级会降低,也就极有可能会被系统给回收掉,有什么好办法呢?Google推荐我们将服务运行到前台,如何知道服务 ...

  2. 14.6.2 Moving or Copying InnoDB Tables to Another Machine 移动或者copy InnoDB 表到另外的机器

    14.6.2 Moving or Copying InnoDB Tables to Another Machine 移动或者copy InnoDB 表到另外的机器 这个章节描述技术关于移动或者copy ...

  3. Linux中more命令的实现

    #include <stdio.h> #include <stdlib.h> #include <string.h> #define PAGELEN 24 #def ...

  4. form表单和表格

    HTML <table> 标签 border pixels 规定表格边框的宽度. STF cellpadding pixels % 规定单元边沿与其内容之间的空白. STF cellspa ...

  5. WebGL自学教程——WebGL演示样本:开始

    最终开始WebGL样品演示,...... 开始 使用WebGL步骤,非常easy: 1. 获得WebGL的渲染环境(也叫渲染上下文). 2. 发挥你的想象力,利用<WebGL參考手冊>中的 ...

  6. URL加随机数的作用

    原文:URL加随机数的作用 大家在系统开发中都可能会在js中用到ajax或者dwr,因为IE的缓存,使得我们在填入相同的值的时候总是使用IE缓存,为了解决这个问题一般可以用一下方法:        1 ...

  7. 普联的路由器TL-WR842N和TL-WR845N还有 TL-WR847N哪一个更好,我是家用

    http://zhidao.baidu.com/question/525244706.html?fr=ala&device=mobile&ssid=0&from=844b&am ...

  8. 解析android framework下利用app_process来调用java写的命令及示例

    解析android framework下利用app_process来调用java写的命令及示例 在android SDK的framework/base/cmds目录下了,有不少目录,这些目的最终都是b ...

  9. 关于索引删除的策略IndexDeletionPolicy

    关于索引删除的策略IndexDeletionPolicy . public IndexWriter(Directory d, Analyzer a, boolean create)          ...

  10. DataGirdView 设置单元格居中

    设置标题行居中: dgvShow.ColumnHeadersDefaultCellStyle.Alignment = DataGridViewContentAlignment.MiddleCenter ...