转自:http://blog.csdn.net/cv_ronny/article/details/17438131

模仿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, OutputArraydst, 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类:

  1. class colorDetect{
  2. private:
  3. int minDist; //minium acceptable distance
  4. Vec3b target;//target color;
  5. Mat result; //the result
  6. public:
  7. colorDetect();
  8. void SetMinDistance(int dist);
  9. void SetTargetColor(uchar red,uchar green,uchar blue);
  10. void SetTargetColor(Vec3b color); //set the target color
  11. Mat process(const Mat& image); //main process
  12. };

其中的minDist是我们定义的阈值用于限定两种颜色之间的距离,相当于PhotoShop中魔术棒工具的阈值。

target是目标颜色,相当于种子颜色。result是存储处理得到的结果。

process是主要的处理程序,下面我们来看process的内容。

  1. Mat colorDetect::process(const Mat& image)
  2. {
  3. Mat ImageLab=image.clone();
  4. result.create(image.rows,image.cols,CV_8U);
  5. //将image转换为Lab格式存储在ImageLab中
  6. cvtColor(image,ImageLab,CV_BGR2Lab);
  7. //将目标颜色由BGR转换为Lab
  8. Mat temp(1,1,CV_8UC3);
  9. temp.at<Vec3b>(0,0)=target;//创建了一张1*1的临时图像并用目标颜色填充
  10. cvtColor(temp,temp,CV_BGR2Lab);
  11. target=temp.at<Vec3b>(0,0);//再从临时图像的Lab格式中取出目标颜色
  12. // 创建处理用的迭代器
  13. Mat_<Vec3b>::iterator it=ImageLab.begin<Vec3b>();
  14. Mat_<Vec3b>::iterator itend=ImageLab.end<Vec3b>();
  15. Mat_<uchar>::iterator itout=result.begin<uchar>();
  16. while(it!=itend)
  17. {
  18. //两个颜色值之间距离的计算
  19. int dist=static_cast<int>(norm<int,3>(Vec3i((*it)[0]-target[0],
  20. (*it)[1]-target[1],(*it)[2]-target[2])));
  21. if(dist<minDist)
  22. (*itout)=255;
  23. else
  24. (*itout)=0;
  25. it++;
  26. itout++;
  27. }
  28. return result;
  29. }

程序中有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);

自己写的测试程序如下:

// learn_opencv3.cpp : 定义控制台应用程序的入口点。
//#include "stdafx.h"
#include "opencv2/core/core.hpp"
#include "opencv2/highgui/highgui.hpp"
#include "opencv2/imgproc/imgproc.hpp"int _tmain(int argc, _TCHAR* argv[])
{cv::Mat image(240, 320, CV_8UC3);cv::Mat image_gray;image_gray.create(image.size(), image.type());image = cv::imread("1.png");cv::cvtColor(image, image_gray, CV_BGR2GRAY);imshow("gray",image);imshow("color", image_gray);cv::waitKey(10000);return 0;
}

OpenCV成长之路(3):模仿PhotoShop中魔术棒工具相关推荐

  1. OpenCV成长之路:模仿PhotoShop中魔术棒工具

    本文的主题实际上是图像的颜色空间的转换,借助一个颜色选取程序来说明OpenCV中颜色转换函数的用法以及一些注意事项. 一.几种常见的颜色空间: RGB颜色空间:RGB采用加法混色法,因为它是描述各种& ...

  2. OpenCV成长之路:直线、轮廓的提取与描述

    http://ronny.blog.51cto.com/8801997/1394139 OpenCV成长之路:直线.轮廓的提取与描述 原创作品,允许转载,转载时请务必以超链接形式标明文章 原始出处 . ...

  3. OpenCV成长之路:图像滤波

    滤波实际上是信号处理里的一个概念,而图像本身也可以看成是一个二维的信号.其中像素点灰度值的高低代表信号的强弱. 高频:图像中灰度变化剧烈的点. 低频:图像中平坦的,灰度变化不大的点. 根据图像的高频与 ...

  4. 漫水填充及Photoshop中魔术棒选择工具的实现

    今天写程序中有一个地方用到了漫水填充(FloodFill).所谓漫水填充,简单来说,如下图中左图,白布上有一块红色的斑点,在这个红色的斑点上点一下,就自动选中了和该点相连的红色的区域,接着将该区域替换 ...

  5. OpenCV精进之路(二十):工具——图像标注小工具

    搞图像深度学习的童鞋一定碰过图像数据标注的东西,当我们训练网络时需要训练集数据,但在网上又没有找到自己想要的数据集,这时候就考虑自己制作自己的数据集了,这时就需要对图像进行标注.图像标注是件很枯燥又很 ...

  6. OpenCV精进之路(十九):工具——程序打包发布

    我们经常遇到这样的需求:我们在VS写好的程序,需要在一个没有装opencv甚至没有装vs的电脑下运行,跑出效果.比如,你在你的电脑用opencv+vs2015写出一个程序,然后老师叫你把程序发给他,他 ...

  7. 从开源小白到 Apache Member,我的成长之路

    2019独角兽企业重金招聘Python工程师标准>>> 我们走过的每一步路,都会留下印记,越坚实,越清晰. 近日,Apache 软件基金会(ASF)官方 Blog 宣布全球新增 40 ...

  8. 《破茧成蝶——用户体验设计师的成长之路》一1.2 邂逅用户体验设计

    本节书摘来自异步社区<破茧成蝶--用户体验设计师的成长之路>一书中的第1章,第1.2节,作者 刘津 , 李月 , 彭文波,更多章节内容可以访问云栖社区"异步社区"公众号 ...

  9. 从开源小白到 Apache Member,阿里技术小哥的成长之路

    我们走过的每一步路,都会留下印记,越坚实,越清晰. 近日,Apache 软件基金会(ASF)官方 Blog 宣布全球新增 40 位 Apache Member,张乎兴有幸成为其中一位. 目前,全球共有 ...

最新文章

  1. .NET Core Community 第二个千星项目诞生:Util
  2. 【Python入门】Python字典的11个方法超级详解
  3. LeetCode 239. Sliding Window Maximum
  4. php中is_int用法,php – is_int()和ctype_digit()之间有区别吗?
  5. 学习python 正则表达式——与你同行!
  6. 下载电子书,给电子书添加总目录页面
  7. go http.Get请求 http.Post请求 http.PostForm请求 Client 超时设置
  8. linux tty驱动名称,Linux下TTY驱动程序分析
  9. 088 菱形继承问题
  10. Windows 2008下Exchange Server部署攻略
  11. Docsify 侧边栏 : 自动生成sidebar与子目录sidebar
  12. (已解决)ubuntu下网易云音乐无法打开
  13. 人民币对美元汇率中间价报6.7969元 下调115个基点
  14. studio3下载安装使用
  15. 怎么将图片压缩到最小,用什么方法
  16. [SHOI2008] 小约翰的游戏
  17. 微信小程序开发架构——JavaScript的基本概述 和 JavaScript在 Nodejs、小程序中、浏览器中的使用方法
  18. linuxprobe 正式开课
  19. 金仓数据库KingbaseES数据库开发指南(4. 面向应用程序的PL/SQL开发)
  20. python 分类变量xgboost_python小白之路:第十九章 XGBoost

热门文章

  1. TF1-项目搭建配置及用户登录
  2. 理解Word2vec
  3. 如何使用缓存提高程序性能
  4. 学习记录:TIM—电容按键检测
  5. 2017⇆2019 AI三年浮沉记
  6. C#获取文件夹/文件的大小以及占用空间 转摘自:http://www.cnblogs.com/chenpeng-dota/articles/2176470.html...
  7. hook xposed 自定义类_使用Xposed实现Hook
  8. 怎样加入第三方客服功能?(百度商桥、美洽、53KF等)
  9. 产品开发中应用计算机的部分,计算机模拟技术在新产品开发及优化铸造工艺中的应用实例.doc...
  10. 光模块之SR、LRM、LR、ER 、ZR对比介绍