OpenCV成长之路(3):模仿PhotoShop中魔术棒工具
本文的主题实际上是图像的颜色空间的转换,借助一个颜色选取程序来说明OpenCV中颜色转换函数的用法以及一些注意事项。
一、几种常见的颜色空间:
RGB颜色空间:RGB采用加法混色法,因为它是描述各种“光”通过何种比例来产生颜色。光线从暗黑开始不断叠加 产生颜色。RGB描述的是红绿蓝三色光的数值。数字图像存储方面一般都是用RGB模式,值得注意的是OpenCV里三通道的存储顺序是BGR。
HSV,HSI:这两个颜色格式都是根据人眼对颜色的区分来定义的格式,其中H(hue)表示色相,S(saturation)表示饱和度,V(value)表示明度,I(intensity)代表了亮度。
Lab空间:模型中均匀改变对应于在感知颜色中的均匀改变,所以我们可以把Lab想像为颜色空间中的一个点,相邻的点靠的越近说明两者的颜色越接近,所以Lab空间常用来度量两个颜色的相似性。
更多颜色空间的知识可以参考:http://en.wikipedia.org/wiki/Color_space
二、OpenCV中的颜色空间转换
OpenCV里通过cvtColor函数来完成图片的颜色转换,cvtColor是在opencv2/imgproc/imgproc.hpp头文件中定义的,它的C++接口如下:
void cvtColor(InputArray src, OutputArray dst, int code, int dstCn=0 )
src:输入图像。
dst:输出图像。
code:颜色转换类型,比如:CV_BGR2Lab,CV_BGR2HSV,CV_HSV2BGR,CV_BGR2RGB。
dstCn:输出图像的通道号,如果默认为0,则表示按输入图像的通道数。
把image图像由BGR转换为Lab:cvtColor(image,image,CV_BGR2Lab)
三、简单的魔术棒程序
首先我们定义一个colorDetect类:
class colorDetect{
private:
int minDist; //minium acceptable distance
Vec3b target;//target color;
Mat result; //the result
public:
colorDetect();
void SetMinDistance(int dist);
void SetTargetColor(uchar red,uchar green,uchar blue);
void SetTargetColor(Vec3b color); //set the target color
Mat process(const Mat& image); //main process
};
其中的minDist是我们定义的阈值用于限定两种颜色之间的距离,相当于PhotoShop中魔术棒工具的阈值。
target是目标颜色,相当于种子颜色。result是存储处理得到的结果。
process是主要的处理程序,下面我们来看process的内容。
Mat colorDetect::process(const Mat& image)
{
Mat ImageLab=image.clone();
result.create(image.rows,image.cols,CV_8U); //将image转换为Lab格式存储在ImageLab中
cvtColor(image,ImageLab,CV_BGR2Lab);
//将目标颜色由BGR转换为Lab
Mat temp(1,1,CV_8UC3);
temp.at<Vec3b>(0,0)=target;//创建了一张1*1的临时图像并用目标颜色填充
cvtColor(temp,temp,CV_BGR2Lab);
target=temp.at<Vec3b>(0,0);//再从临时图像的Lab格式中取出目标颜色 // 创建处理用的迭代器
Mat_<Vec3b>::iterator it=ImageLab.begin<Vec3b>();
Mat_<Vec3b>::iterator itend=ImageLab.end<Vec3b>();
Mat_<uchar>::iterator itout=result.begin<uchar>();
while(it!=itend)
{
//两个颜色值之间距离的计算
int dist=static_cast<int>(norm<int,3>(Vec3i((*it)[0]-target[0],
(*it)[1]-target[1],(*it)[2]-target[2])));
if(dist<minDist)
(*itout)=255;
else
(*itout)=0;
it++;
itout++;
}
return result;
}
程序中有2点需要特别注意:
1,在将图像转换为Lab空间后,目标颜色也需要进行转换,做法是创建了一个临时图像。
2,判断两个颜色之间的距离运算了norm函数,它的运算是norm<typename,dim>(v)。其中v是一个dim维的向量。程序中是一个三维的适量,是两个颜色值两减后的结果。
那值得思考的是能不能把Vec3i((*it)[0]-target[0],(*it)[1]-target[1],(*it)[2]-target[2])替换为Vec3i((*it)-target)呢?答案是否的,因为(*it)-target在实际运算过程中会自动的把相减的结果进行类型限制。
我们对目标颜色和阈值进行这样的设置后可以得到一个示例的效果:
cdet.SetTargetColor(150,150,150);
cdet.SetMinDistance(50);

OpenCV成长之路(3):模仿PhotoShop中魔术棒工具的更多相关文章
- OpenCV成长之路:图像滤波
http://ronny.blog.51cto.com/8801997/1394138 OpenCV成长之路:图像滤波 2014-04-11 14:28:44 标签:opencv 边缘检测 sobel ...
- OpenCV成长之路:图像直方图的应用
OpenCV成长之路:图像直方图的应用 2014-04-11 13:57:03 标签:opencv 图像 直方图 原创作品,允许转载,转载时请务必以超链接形式标明文章 原始出处 .作者信息和本声明.否 ...
- OpenCV成长之路:直线、轮廓的提取与描述
http://ronny.blog.51cto.com/8801997/1394139 OpenCV成长之路:直线.轮廓的提取与描述 原创作品,允许转载,转载时请务必以超链接形式标明文章 原始出处 . ...
- 架构师成长之路1.2-多功能系统信息统计工具dstat
点击返回架构师成长之路 架构师成长之路1.2-多功能系统信息统计工具dstat dstat命令是一个用来替换vmstat.iostat.netstat.nfsstat和ifstat这些命令的工具,是一 ...
- OpenCV成长之路(7):图像滤波
滤波实际上是信号处理里的一个概念,而图像本身也可以看成是一个二维的信号.其中像素点灰度值的高低代表信号的强弱. 高频:图像中灰度变化剧烈的点. 低频:图像中平坦的,灰度变化不大的点. 根据图像的高频与 ...
- OpenCV成长之路 01、图像的读写与显示
一.工具篇 工欲善其事,必先利其器.学习OpenCV,肯定少不于基本的编程工具与OpenCV库.在Windows平台下你可以选择Visual Studio.CodeBlock等,当然你也可以选择在Li ...
- OpenCV成长之路(2):图像的遍历
我们在实际应用中对图像进行的操作,往往并不是将图像作为一个整体进行操作,而是对图像中的所有点或特殊点进行运算,所以遍历图像就显得很重要,如何高效的遍历图像是一个很值得探讨的问题. 一.遍历图像的4种方 ...
- OpenCV成长之路(10):视频的处理
视频中包含的信息量要远远大于图片,对视频的处理分析也越来越成为计算机视觉的主流,而本质上视频是由一帧帧的图像组成,所以视频处理最终还是要归结于图像处理,但在视频处理中,有更多的时间维的信息可以利用.本 ...
- OpenCV成长之路(9):特征点检测与图像匹配
特征点又称兴趣点.关键点,它是在图像中突出且具有代表意义的一些点,通过这些点我们可以用来识别图像.进行图像配准.进行3D重建等.本文主要介绍OpenCV中几种定位与表示关键点的函数. 一.Harris ...
随机推荐
- Javascript权威指南——第一章Javascript概述
示例:javascript贷款计算器 相关技术: 1.如何在文档中查找元素: 2.如何通过表单input元素来获取用户的输入数据: 3.如何通过文档元素来设置HTML内容: 4.如何将数据存储在浏览器 ...
- 解决问题--VS2012中一个Panel覆盖另一个Panel时拖动时容易造成两个控件成父子关系的避免
在*.Designer.cs中,假如想把panel1覆盖到panel2上,但是VS自动让panel1成为panel2的子控件了,在文件中会有this.panel2.Controls.Add(this. ...
- [Redis]通过代码配置Redis
查看了文档https://azure.microsoft.com/en-us/documentation/articles/cache-how-to-scale/,发现可以使用代码来配置Redis,所 ...
- js控制网页滚动条往下滚动
function aa(i){ var tm = setInterval(function(){ var t = $(window).scrollTop(); , -) : Math.max((i-t ...
- reactjs 注意点
render的return return前要留一空行 return的括号要分别各占一行,不能与html同行 return中的html必须要有顶层容器包裹 return中的循环不能用for,改用map方 ...
- 让Xcode的控制台支持LLDB类型的打印
这个技巧个人认为非常有用 当Xcode在断点调试的时候,在控制台中输入 po self.view.frame 类似这样的命令会挂掉,不信可以亲自去试试(Xcode7 以后支持LLDB类型的打印) 那么 ...
- vi和vim区别及命令详解
vi和vim都是Linux中的编辑器,不同的是vim比较高级,可以视为vi的升级版本.vi使用于文本编辑,但是vim更适用于coding. 现将vim的命令行收集于下: vi有3个模式:插入模 ...
- Javabean+servlet+JSP(html)实例应用
大家都知道Javabean+servlet+JSP是最简单的MVC模式.的确,在一个小型的项目中,这个模式完全够用. 它优雅并且简洁.加上jQueryui的完美展示效果,让这个模式看起来非常合适.当然 ...
- Sublime Text 2 快捷键 (windows)
转自:http://istyles.blog.163.com/blog/static/1811003892011828111418654/ Lucifr翻译了 Sublime Text 2 快捷键 M ...
- 安装好Android Studio(如果内存足够可以改下as的内存)
找到studio的bin目录,找到如下文件