在本教程中,我们将学习计算机视觉中常用的颜色空间,并使用它进行基于颜色的分割。我们还将共享c++和Python的演示代码。

我们在许多计算机视觉应用中都遇到了这个问题,包括肤色检测、交通灯识别等基于颜色的分割。让我们看看如何帮助他为他的机器人建立一个强大的颜色检测系统。

文章组织如下:

  • 首先,我们将看到如何读取OpenCV中的图像,并将其转换为不同的颜色空间,并看看每个颜色空间的不同通道为我们提供了什么新信息。
  • 我们将采用一种简单的颜色分割算法,并仔细考虑其缺点。
  • 然后我们将进入一些分析,并使用系统的方法进行选择:(1)正确的颜色空间。(2)正确的分割阈值。

1.不同的颜色空间

在本节中,我们将介绍计算机视觉中使用的一些重要的颜色空间。我们不会描述它们背后的理论,因为它可以在维基百科上找到。相反,我们将建立一个基本的直觉,并学习一些重要的性质,这将对以后的决策有用。

让我们加载两个相同立方体的图像。默认情况下以BGR格式加载。我们可以使用OpenCV函数cvtColor()在不同的颜色空间之间进行转换,稍后会展示。

第一幅图像是在阳光明媚的室外条件下拍摄的,第二幅是在室内正常光照条件下拍摄的。

(1)RGB颜色空间

RGB颜色空间具有以下属性:

  • 它是一个加性色彩空间,其中颜色由红色、绿色和蓝色值的线性组合获得。
  • 这三个通道是由照射到表面的光线量关联起来的。
    让我们将这两幅图像分割成R、G和B分量,并观察它们,以更深入地了解颜色空间。

    观测
    如果你看蓝色通道,可以看到,在室内光照条件下,第二幅图像中的蓝色和白色片段看起来很相似,但在第一幅图像中有明显的区别。这种不均匀性使得基于颜色的分割在这个颜色空间中非常困难。此外,两幅图像的值之间存在整体差异。下面我们总结了RGB颜色空间的内在问题:
  • 重要的知觉不均匀
  • 混合色度(颜色相关信息)和亮度(强度相关信息)数据。

(2)LAB色彩空间

LAB颜色空间有三个组成部分。

  • L:亮度(强度)。
  • a:颜色成分从绿色到品红不等。
  • b:颜色成分范围从蓝色到黄色。

LAB颜色空间与RGB颜色空间有很大的不同。在RGB颜色空间中,颜色信息被分为三个通道,但同样的三个通道也编码亮度信息。另一方面,在Lab颜色空间中,L通道与颜色信息无关,只对亮度进行编码。另外两个通道编码颜色。

它具有以下属性。

  • 感知上一致的颜色空间,近似于我们感知颜色的方式。
  • 独立于设备(捕获或显示)。
  • 在Adobe Photoshop中广泛使用。
  • 与RGB颜色空间有关的是一个复变换方程。


观测

  • 从图中可以很清楚的看出,光照的变化对L分量的影响最大。
  • 包含颜色信息的A和B分量没有发生大的变化。
  • 室内的颜色信息与室外的颜色信息相似

(3)YCrCb彩色空间

YCrCb颜色空间是由RGB颜色空间派生而来的,它有以下三个组成部分。

  • Y -经过伽玛校正后由RGB得到的亮度或亮度分量。
  • Cr = R - Y(红色分量离Luma有多远)。
  • Cb = B - Y(蓝色分量到Luma的距离)。

这个颜色空间具有以下属性。

  • 将亮度和色度组件分离到不同的通道中。
  • 主要用于电视传输压缩(Cr和Cb元件)。
  • 设备相关

    观测
  • 亮度和颜色成分的光照变化,与LAB类似的观测结果。
  • 与LAB相比,在室外图像中,红色和橙色的感知差异更小。
  • 白色在所有三种成分中都发生了变化。

(4)HSV彩色空间

HSV颜色空间有以下三个组成部分:

  • H -色相(主波长)。
  • S -饱和度(纯度/颜色的深浅)。
  • V -值(强度)。

让我们列举它的一些属性。

  • 最好的是,它只使用一个通道来描述颜色(H),使它非常直观地指定颜色。
  • 设备相关


观测

  • 两幅图像的H分量非常相似,这表明即使在光照变化的情况下,颜色信息也是完整的。
  • 在两幅图像中S分量也非常相似。
  • V分量捕捉落在它上面的光量,因此它会随着照明的变化而变化。
  • 室外图像和室内图像的红色块值有很大的差异。这是因为Hue被表示为一个圆,红色是在起始角度。因此,它可以取[300,360]和[0,60]之间的值。

2.如何使用这些颜色空间进行分割

(1)最简单地方式

现在我们已经对不同的颜色空间有了一些了解,让我们首先尝试使用它们来检测立方体中的绿色。

  • 步骤1:获取特定颜色的颜色值
    找出每个颜色空间的绿色值的近似范围。为此,使用一个交互式GUI,你可以通过将鼠标悬停在图像上来检查每个像素的所有颜色空间的值,如下所示:

交互式GUI代码如下:

(1)Python
interactiveColorDetect.py

import cv2,argparse,glob
import numpy as np# 鼠标回调函数
def showPixelValue(event,x,y,flags,param):global img, combinedResult, placeholderif event == cv2.EVENT_MOUSEMOVE:# 从鼠标在(x,y)中的位置获取像素值bgr = img[y,x]# 将BGR像素转换为其他颜色格式ycb = cv2.cvtColor(np.uint8([[bgr]]),cv2.COLOR_BGR2YCrCb)[0][0]lab = cv2.cvtColor(np.uint8([[bgr]]),cv2.COLOR_BGR2Lab)[0][0]hsv = cv2.cvtColor(np.uint8([[bgr]]),cv2.COLOR_BGR2HSV)[0][0]# 创建一个空占位符来显示值placeholder = np.zeros((img.shape[0],400,3),dtype=np.uint8)# 用颜色空间的值填充占位符cv2.putText(placeholder, "BGR {}".format(bgr), (20, 70), cv2.FONT_HERSHEY_COMPLEX, .9, (255,255,255), 1, cv2.LINE_AA)cv2.putText(placeholder, "HSV {}".format(hsv), (20, 140), cv2.FONT_HERSHEY_COMPLEX, .9, (255,255,255), 1, cv2.LINE_AA)cv2.putText(placeholder, "YCrCb {}".format(ycb), (20, 210), cv2.FONT_HERSHEY_COMPLEX, .9, (255,255,255), 1, cv2.LINE_AA)cv2.putText(placeholder, "LAB {}".format(lab), (20, 280), cv2.FONT_HERSHEY_COMPLEX, .9, (255,255,255), 1, cv2.LINE_AA)# 将两个结果合并在一个图像中并排显示combinedResult = np.hstack([img,placeholder])cv2.imshow('PRESS P for Previous, N for Next Image',combinedResult)if __name__ == '__main__' :# 加载图像并设置鼠标回调函数global imgfiles = glob.glob('images/rub*.jpg')files.sort()img = cv2.imread(files[0])img = cv2.resize(img,(400,400))cv2.imshow('PRESS P for Previous, N for Next Image',img)# 创建一个空窗口cv2.namedWindow('PRESS P for Previous, N for Next Image')# 为鼠标上的任何事件创建一个回调函数cv2.setMouseCallback('PRESS P for Previous, N for Next Image',showPixelValue)i = 0while(1):k = cv2.waitKey(1) & 0xFF# 检查文件夹中的下一个图像if k == ord('n'):i += 1img = cv2.imread(files[i%len(files)])img = cv2.resize(img,(400,400))cv2.imshow('PRESS P for Previous, N for Next Image',img)# 检查文件夹中的前一个图像elif k == ord('p'):i -= 1img = cv2.imread(files[i%len(files)])img = cv2.resize(img,(400,400))cv2.imshow('PRESS P for Previous, N for Next Image',img)elif k == 27:cv2.destroyAllWindows()break

(2) C++
interactiveColorDetect.cpp

#include "opencv2/opencv.hpp"
#include <iostream>
using namespace cv;
using namespace std;//全局变量
Mat img, placeholder;// 鼠标上的任何事件的回调函数
void onMouse( int event, int x, int y, int flags, void* userdata )
{   if( event == EVENT_MOUSEMOVE ){Vec3b bgrPixel(img.at<Vec3b>(y, x));Mat3b hsv,ycb,lab;// 从向量创建 Mat 对象,因为 cvtColor 接受一个 Mat 对象Mat3b bgr (bgrPixel);//将单个像素的BGR Mat转换为其他格式cvtColor(bgr, ycb, COLOR_BGR2YCrCb);cvtColor(bgr, hsv, COLOR_BGR2HSV);cvtColor(bgr, lab, COLOR_BGR2Lab);//从 Mat 取回向量Vec3b hsvPixel(hsv.at<Vec3b>(0,0));Vec3b ycbPixel(ycb.at<Vec3b>(0,0));Vec3b labPixel(lab.at<Vec3b>(0,0));// 创建一个空的占位符来显示值placeholder = Mat::zeros(img.rows,400,CV_8UC3);//用颜色空间的值填充占位符putText(placeholder, format("BGR [%d, %d, %d]",bgrPixel[0],bgrPixel[1],bgrPixel[2]), Point(20, 70), FONT_HERSHEY_COMPLEX, .9, Scalar(255,255,255), 1);putText(placeholder, format("HSV [%d, %d, %d]",hsvPixel[0],hsvPixel[1],hsvPixel[2]), Point(20, 140), FONT_HERSHEY_COMPLEX, .9, Scalar(255,255,255), 1);putText(placeholder, format("YCrCb [%d, %d, %d]",ycbPixel[0],ycbPixel[1],ycbPixel[2]), Point(20, 210), FONT_HERSHEY_COMPLEX, .9, Scalar(255,255,255), 1);putText(placeholder, format("LAB [%d, %d, %d]",labPixel[0],labPixel[1],labPixel[2]), Point(20, 280), FONT_HERSHEY_COMPLEX, .9, Scalar(255,255,255), 1);Size sz1 = img.size();Size sz2 = placeholder.size();//将两个结果合并在一个图像中并排显示Mat combinedResult(sz1.height, sz1.width+sz2.width, CV_8UC3);Mat left(combinedResult, Rect(0, 0, sz1.width, sz1.height));img.copyTo(left);Mat right(combinedResult, Rect(sz1.width, 0, sz2.width, sz2.height));placeholder.copyTo(right);imshow("PRESS P for Previous, N for Next Image", combinedResult);}
}int main( int argc, const char** argv )
{// 读取输入图像int image_number = 0;int nImages = 10;if(argc > 1)nImages = atoi(argv[1]);char filename[20];sprintf(filename,"images/rub%02d.jpg",image_number%nImages);img = imread(filename);// 将图像大小调整为 400x400Size rsize(400,400);resize(img,img,rsize);if(img.empty()){return -1;}// 创建一个空窗口namedWindow("PRESS P for Previous, N for Next Image", WINDOW_AUTOSIZE);   // 为鼠标上的任何事件创建一个回调函数setMouseCallback( "PRESS P for Previous, N for Next Image", onMouse );imshow( "PRESS P for Previous, N for Next Image", img );while(1){char k = waitKey(1) & 0xFF;if (k == 27)break;//检查文件夹中的下一个图像if (k =='n'){image_number++;sprintf(filename,"images/rub%02d.jpg",image_number%nImages);img = imread(filename);resize(img,img,rsize); }//检查文件夹中先前的图像else if (k =='p'){image_number--;sprintf(filename,"images/rub%02d.jpg",image_number%nImages);img = imread(filename);resize(img,img,rsize);}}return 0;
}
  • 步骤2:应用阈值进行分割
    从图像中提取出与绿色像素值相近的所有像素。我们可以为每个颜色空间取一个+/- 40的范围,并检查结果看起来如何。我们将使用opencv函数inRange()来查找绿色像素的mask,然后使用bitwise_and()操作使用mask从图像中获取绿色像素。
    interactiveColorSegment.py
    (1)Python
import cv2,time,argparse,glob
import numpy as np# 要跟踪的全局变量
show = Falsedef onTrackbarActivity(x):global showshow = Truepassif __name__ == '__main__' :# 获取文件名files = glob.glob('images/rub*.jpg')files.sort()# 加载图片original = cv2.imread(files[0])# 调整图像大小rsize = 250original = cv2.resize(original,(rsize,rsize))# 屏幕上窗口开始的位置initialX = 50initialY = 50# 创建显示图像的窗口cv2.namedWindow('P-> Previous, N-> Next',cv2.WINDOW_AUTOSIZE)cv2.namedWindow('SelectBGR',cv2.WINDOW_AUTOSIZE)cv2.namedWindow('SelectHSV',cv2.WINDOW_AUTOSIZE)cv2.namedWindow('SelectYCB',cv2.WINDOW_AUTOSIZE)cv2.namedWindow('SelectLAB',cv2.WINDOW_AUTOSIZE)# 移动窗口,将它们水平堆叠cv2.moveWindow('P-> Previous, N-> Next',initialX,initialY)cv2.moveWindow('SelectBGR',initialX + (rsize + 5),initialY)cv2.moveWindow('SelectHSV',initialX + 2*(rsize + 5),initialY)cv2.moveWindow('SelectYCB',initialX + 3*(rsize + 5),initialY)cv2.moveWindow('SelectLAB',initialX + 4*(rsize + 5),initialY)#创建跟踪条以获取YCrCb的值cv2.createTrackbar('CrMin','SelectYCB',0,255,onTrackbarActivity)cv2.createTrackbar('CrMax','SelectYCB',0,255,onTrackbarActivity)cv2.createTrackbar('CbMin','SelectYCB',0,255,onTrackbarActivity)cv2.createTrackbar('CbMax','SelectYCB',0,255,onTrackbarActivity)cv2.createTrackbar('YMin','SelectYCB',0,255,onTrackbarActivity)cv2.createTrackbar('YMax','SelectYCB',0,255,onTrackbarActivity)#创建跟踪条来获取HSV的值cv2.createTrackbar('HMin','SelectHSV',0,180,onTrackbarActivity)cv2.createTrackbar('HMax','SelectHSV',0,180,onTrackbarActivity)cv2.createTrackbar('SMin','SelectHSV',0,255,onTrackbarActivity)cv2.createTrackbar('SMax','SelectHSV',0,255,onTrackbarActivity)cv2.createTrackbar('VMin','SelectHSV',0,255,onTrackbarActivity)cv2.createTrackbar('VMax','SelectHSV',0,255,onTrackbarActivity)#创建跟踪条来获取BGR的值cv2.createTrackbar('BGRBMin','SelectBGR',0,255,onTrackbarActivity)cv2.createTrackbar('BGRBMax','SelectBGR',0,255,onTrackbarActivity)cv2.createTrackbar('BGRGMin','SelectBGR',0,255,onTrackbarActivity)cv2.createTrackbar('BGRGMax','SelectBGR',0,255,onTrackbarActivity)cv2.createTrackbar('BGRRMin','SelectBGR',0,255,onTrackbarActivity)cv2.createTrackbar('BGRRMax','SelectBGR',0,255,onTrackbarActivity)#创建跟踪条来获取LAB的值cv2.createTrackbar('LABLMin','SelectLAB',0,255,onTrackbarActivity)cv2.createTrackbar('LABLMax','SelectLAB',0,255,onTrackbarActivity)cv2.createTrackbar('LABAMin','SelectLAB',0,255,onTrackbarActivity)cv2.createTrackbar('LABAMax','SelectLAB',0,255,onTrackbarActivity)cv2.createTrackbar('LABBMin','SelectLAB',0,255,onTrackbarActivity)cv2.createTrackbar('LABBMax','SelectLAB',0,255,onTrackbarActivity)# 初始显示所有图像cv2.imshow('SelectHSV',original)cv2.imshow('SelectYCB',original)cv2.imshow('SelectLAB',original)cv2.imshow('SelectBGR',original)i = 0while(1):cv2.imshow('P-> Previous, N-> Next',original)  k = cv2.waitKey(1) & 0xFF# 检查文件夹中的下一个图像    if k == ord('n'):i += 1original = cv2.imread(files[i%len(files)])original = cv2.resize(original,(rsize,rsize))show = True# 检查文件夹中的先前图像    elif k == ord('p'):i -= 1original = cv2.imread(files[i%len(files)])original = cv2.resize(original,(rsize,rsize))show = True# 按下esc键后关闭所有窗口elif k == 27:breakif show: # 如果在轨迹栏上有任何事件show = False# 从BGR跟踪器获取值BMin = cv2.getTrackbarPos('BGRBMin','SelectBGR')GMin = cv2.getTrackbarPos('BGRGMin','SelectBGR')RMin = cv2.getTrackbarPos('BGRRMin','SelectBGR')BMax = cv2.getTrackbarPos('BGRBMax','SelectBGR')GMax = cv2.getTrackbarPos('BGRGMax','SelectBGR')RMax = cv2.getTrackbarPos('BGRRMax','SelectBGR')minBGR = np.array([BMin, GMin, RMin])maxBGR = np.array([BMax, GMax, RMax])# 从HSV跟踪器获取值HMin = cv2.getTrackbarPos('HMin','SelectHSV')SMin = cv2.getTrackbarPos('SMin','SelectHSV')VMin = cv2.getTrackbarPos('VMin','SelectHSV')HMax = cv2.getTrackbarPos('HMax','SelectHSV')SMax = cv2.getTrackbarPos('SMax','SelectHSV')VMax = cv2.getTrackbarPos('VMax','SelectHSV')minHSV = np.array([HMin, SMin, VMin])maxHSV = np.array([HMax, SMax, VMax])# 从LAB跟踪器获取值LMin = cv2.getTrackbarPos('LABLMin','SelectLAB')AMin = cv2.getTrackbarPos('LABAMin','SelectLAB')BMin = cv2.getTrackbarPos('LABBMin','SelectLAB')LMax = cv2.getTrackbarPos('LABLMax','SelectLAB')AMax = cv2.getTrackbarPos('LABAMax','SelectLAB')BMax = cv2.getTrackbarPos('LABBMax','SelectLAB')minLAB = np.array([LMin, AMin, BMin])maxLAB = np.array([LMax, AMax, BMax])# 从YCrCb跟踪器获取值YMin = cv2.getTrackbarPos('YMin','SelectYCB')CrMin = cv2.getTrackbarPos('CrMin','SelectYCB')CbMin = cv2.getTrackbarPos('CbMin','SelectYCB')YMax = cv2.getTrackbarPos('YMax','SelectYCB')CrMax = cv2.getTrackbarPos('CrMax','SelectYCB')CbMax = cv2.getTrackbarPos('CbMax','SelectYCB')minYCB = np.array([YMin, CrMin, CbMin])maxYCB = np.array([YMax, CrMax, CbMax])# 将BGR图像转换为其他颜色空间imageBGR = np.copy(original)imageHSV = cv2.cvtColor(original,cv2.COLOR_BGR2HSV)imageYCB = cv2.cvtColor(original,cv2.COLOR_BGR2YCrCb)imageLAB = cv2.cvtColor(original,cv2.COLOR_BGR2LAB)# 使用从轨迹栏获得的最小值和最大值创建掩码,并应用bitewise_and操作来获得结果        maskBGR = cv2.inRange(imageBGR,minBGR,maxBGR)resultBGR = cv2.bitwise_and(original, original, mask = maskBGR)         maskHSV = cv2.inRange(imageHSV,minHSV,maxHSV)resultHSV = cv2.bitwise_and(original, original, mask = maskHSV)maskYCB = cv2.inRange(imageYCB,minYCB,maxYCB)resultYCB = cv2.bitwise_and(original, original, mask = maskYCB)         maskLAB = cv2.inRange(imageLAB,minLAB,maxLAB)resultLAB = cv2.bitwise_and(original, original, mask = maskLAB)         # 显示结果cv2.imshow('SelectBGR',resultBGR)cv2.imshow('SelectYCB',resultYCB)cv2.imshow('SelectLAB',resultLAB)cv2.imshow('SelectHSV',resultHSV)cv2.destroyAllWindows()

(2) C++
interactiveColorSegment.cpp

#include "opencv2/opencv.hpp"
#include <iostream>
#include <cstring>using namespace cv;
using namespace std;
// 要跟踪的全局变量
bool show = false;// 为跟踪栏上的事件创建回调
void onTrackbarActivity(int pos, void* userdata){// 只需要在全局变量中指定有一个事件show = true;return;
}int main(int argc, char **argv)
{int image_number = 0;int nImages = 10;if(argc > 1)nImages = atoi(argv[1]);char filename[20];sprintf(filename,"images/rub%02d.jpg",image_number%nImages);Mat original = imread(filename);// 图像调整宽度和高度int resizeHeight = 250;int resizeWidth = 250;Size rsize(resizeHeight,resizeWidth);resize(original, original, rsize);// 屏幕上窗口开始的位置int initialX = 50;int initialY = 50;// 创建显示图像的窗口namedWindow("P-> Previous, N-> Next", WINDOW_AUTOSIZE);namedWindow("SelectBGR", WINDOW_AUTOSIZE);namedWindow("SelectHSV", WINDOW_AUTOSIZE);namedWindow("SelectYCB", WINDOW_AUTOSIZE);namedWindow("SelectLAB", WINDOW_AUTOSIZE);// 移动窗口,将它们水平堆叠 moveWindow("P-> Previous, N-> Next", initialX, initialY);moveWindow("SelectBGR", initialX + 1 * (resizeWidth + 5), initialY);moveWindow("SelectHSV", initialX + 2 * (resizeWidth + 5), initialY);moveWindow("SelectYCB", initialX + 3 * (resizeWidth + 5), initialY);moveWindow("SelectLAB", initialX + 4 * (resizeWidth + 5), initialY);// 创建跟踪条以获取YCrCb的值createTrackbar("CrMin", "SelectYCB", 0, 255, onTrackbarActivity);createTrackbar("CrMax", "SelectYCB", 0, 255, onTrackbarActivity);createTrackbar("CbMin", "SelectYCB", 0, 255, onTrackbarActivity);createTrackbar("CbMax", "SelectYCB", 0, 255, onTrackbarActivity);createTrackbar("YMin", "SelectYCB", 0, 255, onTrackbarActivity);createTrackbar("YMax", "SelectYCB", 0, 255, onTrackbarActivity);// 创建跟踪条来获取HSV的值 createTrackbar("HMin", "SelectHSV", 0, 180, onTrackbarActivity);createTrackbar("HMax", "SelectHSV", 0, 180, onTrackbarActivity);createTrackbar("SMin", "SelectHSV", 0, 255, onTrackbarActivity);createTrackbar("SMax", "SelectHSV", 0, 255, onTrackbarActivity);createTrackbar("VMin", "SelectHSV", 0, 255, onTrackbarActivity);createTrackbar("VMax", "SelectHSV", 0, 255, onTrackbarActivity);// 创建跟踪条来获取BGR的值createTrackbar("BMin", "SelectBGR", 0, 255, onTrackbarActivity);createTrackbar("BMax", "SelectBGR", 0, 255, onTrackbarActivity);createTrackbar("GMin", "SelectBGR", 0, 255, onTrackbarActivity);createTrackbar("GMax", "SelectBGR", 0, 255, onTrackbarActivity);createTrackbar("RMin", "SelectBGR", 0, 255, onTrackbarActivity);createTrackbar("RMax", "SelectBGR", 0, 255, onTrackbarActivity);// 创建跟踪条来获取LAB的值createTrackbar("LMin", "SelectLAB", 0, 255, onTrackbarActivity);createTrackbar("LMax", "SelectLAB", 0, 255, onTrackbarActivity);createTrackbar("AMin", "SelectLAB", 0, 255, onTrackbarActivity);createTrackbar("AMax", "SelectLAB", 0, 255, onTrackbarActivity);createTrackbar("BMin", "SelectLAB", 0, 255, onTrackbarActivity);createTrackbar("BMax", "SelectLAB", 0, 255, onTrackbarActivity);// 初始显示所有图像imshow("SelectHSV", original);imshow("SelectYCB", original);imshow("SelectLAB", original);imshow("SelectBGR", original);//声明局部变量int BMin, GMin, RMin;int BMax, GMax, RMax;Scalar minBGR, maxBGR;int HMin, SMin, VMin;int HMax, SMax, VMax;Scalar minHSV, maxHSV;int LMin, aMin, bMin;int LMax, aMax, bMax;Scalar minLab, maxLab;int YMin, CrMin, CbMin;int YMax, CrMax, CbMax;Scalar minYCrCb, maxYCrCb;Mat imageBGR, imageHSV, imageLab, imageYCrCb;Mat maskBGR, maskHSV, maskLab, maskYCrCb;Mat resultBGR, resultHSV, resultLab, resultYCrCb;char k;while (1){imshow("P-> Previous, N-> Next", original);k = waitKey(1) & 0xFF;//检查文件夹中的下一个图像if (k =='n'){image_number++;sprintf(filename,"images/rub%02d.jpg",image_number%nImages);original = imread(filename);resize(original,original,rsize); show = true;}//检查文件夹中先前的图像else if (k =='p'){image_number--;sprintf(filename,"images/rub%02d.jpg",image_number%nImages);original = imread(filename);resize(original,original,rsize);show = true;}// 按下esc键后关闭所有窗口     if (k == 27){break;}if (show) { //如果在轨迹栏上有任何事件show = false;// 从BGR trackbar获取值BMin = getTrackbarPos("BMin", "SelectBGR");GMin = getTrackbarPos("GMin", "SelectBGR");RMin = getTrackbarPos("RMin", "SelectBGR");BMax = getTrackbarPos("BMax", "SelectBGR");GMax = getTrackbarPos("GMax", "SelectBGR");RMax = getTrackbarPos("RMax", "SelectBGR");minBGR = Scalar(BMin, GMin, RMin);maxBGR = Scalar(BMax, GMax, RMax);// 从HSV轨迹栏获取值HMin = getTrackbarPos("HMin", "SelectHSV");SMin = getTrackbarPos("SMin", "SelectHSV");VMin = getTrackbarPos("VMin", "SelectHSV");HMax = getTrackbarPos("HMax", "SelectHSV");SMax = getTrackbarPos("SMax", "SelectHSV");VMax = getTrackbarPos("VMax", "SelectHSV");minHSV = Scalar(HMin, SMin, VMin);maxHSV = Scalar(HMax, SMax, VMax);// 从LAB跟踪栏获取值LMin = getTrackbarPos("LMin", "SelectLAB");aMin = getTrackbarPos("AMin", "SelectLAB");bMin = getTrackbarPos("BMin", "SelectLAB");LMax = getTrackbarPos("LMax", "SelectLAB");aMax = getTrackbarPos("AMax", "SelectLAB");bMax = getTrackbarPos("BMax", "SelectLAB");minLab = Scalar(LMin, aMin, bMin);maxLab = Scalar(LMax, aMax, bMax);// 从YCrCb轨迹栏获取值YMin = getTrackbarPos("YMin", "SelectYCB");CrMin = getTrackbarPos("CrMin", "SelectYCB");CbMin = getTrackbarPos("CbMin", "SelectYCB");YMax = getTrackbarPos("YMax", "SelectYCB");CrMax = getTrackbarPos("CrMax", "SelectYCB");CbMax = getTrackbarPos("CbMax", "SelectYCB");minYCrCb = Scalar(YMin, CrMin, CbMin);maxYCrCb = Scalar(YMax, CrMax, CbMax);// 将BGR图像转换为其他颜色空间original.copyTo(imageBGR);cvtColor(original, imageHSV, COLOR_BGR2HSV);cvtColor(original, imageYCrCb, COLOR_BGR2YCrCb);cvtColor(original, imageLab, COLOR_BGR2Lab);// 使用从轨迹栏获得的最小值和最大值创建掩码,并应用bitewise_and操作来获得结果        inRange(imageBGR, minBGR, maxBGR, maskBGR);resultBGR = Mat::zeros(original.rows, original.cols, CV_8UC3);bitwise_and(original, original, resultBGR, maskBGR);inRange(imageHSV, minHSV, maxHSV, maskHSV);resultHSV = Mat::zeros(original.rows, original.cols, CV_8UC3);bitwise_and(original, original, resultHSV, maskHSV);inRange(imageYCrCb, minYCrCb, maxYCrCb, maskYCrCb);resultYCrCb = Mat::zeros(original.rows, original.cols, CV_8UC3);bitwise_and(original, original, resultYCrCb, maskYCrCb);inRange(imageLab, minLab, maxLab, maskLab);resultLab = Mat::zeros(original.rows, original.cols, CV_8UC3);bitwise_and(original, original, resultLab, maskLab);// 结果显示imshow("SelectBGR", resultBGR);imshow("SelectYCB", resultYCrCb);imshow("SelectLAB", resultLab);imshow("SelectHSV", resultHSV);}}destroyAllWindows();return 0;
}

(2)为更好的解决方案进行一些数据分析

  • 第一步:数据收集
    我收集了立方体在不同光照条件下的10张图像,分别裁剪每一种颜色,得到6个不同颜色的数据集。你可以从视觉上看到颜色发生了多大的变化。
  • 步骤2:计算密度图
    检查特定颜色的分布,比如蓝色或黄色在不同颜色空间中的分布。密度图或2D直方图给出了关于给定颜色值变化的想法。例如,理想情况下,蓝色图像的蓝色通道的值应该始终为255。但实际上,它分布在0到255之间。
# 导入所需的包
import matplotlib.pyplot as plt
from matplotlib.colors import LogNorm
import cv2, glob
import numpy as np# 指定要绘制的直方图的颜色
color = 'pieces/yellow'
#图是否应该是完整的还是缩小的
zoom = 1
# 加载文件夹中的所有文件
files = glob.glob(color + '*.jpg')
files.sort()
# 空数组用于分离的通道
B = np.array([])
G = np.array([])
R = np.array([])
H = np.array([])
S = np.array([])
V = np.array([])
Y = np.array([])
Cr = np.array([])
Cb = np.array([])
LL = np.array([])
LA = np.array([])
LB = np.array([])# 数据创建
# 将每个文件中的值附加到各自的通道中
for fi in files[:]:# BGRim = cv2.imread(fi)b = im[:, :, 0]b = b.reshape(b.shape[0] * b.shape[1])g = im[:, :, 1]g = g.reshape(g.shape[0] * g.shape[1])r = im[:, :, 2]r = r.reshape(r.shape[0] * r.shape[1])B = np.append(B, b)G = np.append(G, g)R = np.append(R, r)# HSVhsv = cv2.cvtColor(im, cv2.COLOR_BGR2HSV)h = hsv[:, :, 0]h = h.reshape(h.shape[0] * h.shape[1])s = hsv[:, :, 1]s = s.reshape(s.shape[0] * s.shape[1])v = hsv[:, :, 2]v = v.reshape(v.shape[0] * v.shape[1])H = np.append(H, h)S = np.append(S, s)V = np.append(V, v)# YCrCbycb = cv2.cvtColor(im, cv2.COLOR_BGR2YCrCb)y = ycb[:, :, 0]y = y.reshape(y.shape[0] * y.shape[1])cr = ycb[:, :, 1]cr = cr.reshape(cr.shape[0] * cr.shape[1])cb = ycb[:, :, 2]cb = cb.reshape(cb.shape[0] * cb.shape[1])Y = np.append(Y, y)Cr = np.append(Cr, cr)Cb = np.append(Cb, cb)# Lablab = cv2.cvtColor(im, cv2.COLOR_BGR2LAB)ll = lab[:, :, 0]ll = ll.reshape(ll.shape[0] * ll.shape[1])la = lab[:, :, 1]la = la.reshape(la.shape[0] * la.shape[1])lb = lab[:, :, 2]lb = lb.reshape(lb.shape[0] * lb.shape[1])LL = np.append(LL, ll)LA = np.append(LA, la)LB = np.append(LB, lb)# 绘制柱状图
nbins = 10
plt.figure(figsize=[20, 10])
plt.subplot(2, 3, 1)
plt.hist2d(B, G, bins=nbins, norm=LogNorm())
plt.xlabel('B')
plt.ylabel('G')
plt.title('RGB')
if not zoom:plt.xlim([0, 255])plt.ylim([0, 255])
plt.colorbar()
plt.subplot(2, 3, 2)
plt.hist2d(B, R, bins=nbins, norm=LogNorm())
plt.colorbar()
plt.xlabel('B')
plt.ylabel('R')
plt.title('RGB')
if not zoom:plt.xlim([0, 255])plt.ylim([0, 255])
plt.subplot(2, 3, 3)
plt.hist2d(R, G, bins=nbins, norm=LogNorm())
plt.colorbar()
plt.xlabel('R')
plt.ylabel('G')
plt.title('RGB')
if not zoom:plt.xlim([0, 255])plt.ylim([0, 255])plt.subplot(2, 3, 4)
plt.hist2d(H, S, bins=nbins, norm=LogNorm())
plt.colorbar()
plt.xlabel('H')
plt.ylabel('S')
plt.title('HSV')
if not zoom:plt.xlim([0, 180])plt.ylim([0, 255])plt.subplot(2, 3, 5)
plt.hist2d(Cr, Cb, bins=nbins, norm=LogNorm())
plt.colorbar()
plt.xlabel('Cr')
plt.ylabel('Cb')
plt.title('YCrCb')
if not zoom:plt.xlim([0, 255])plt.ylim([0, 255])plt.subplot(2, 3, 6)
plt.hist2d(LA, LB, bins=nbins, norm=LogNorm())
plt.colorbar()
plt.xlabel('A')
plt.ylabel('B')
plt.title('LAB')
if not zoom:plt.xlim([0, 255])plt.ylim([0, 255])plt.savefig(color + '.png', bbox_inches='tight')
else:plt.savefig(color + '-zoom.png', bbox_inches='tight')plt.show()

当分析blue图片时候:

当分析green图片时候:

当分析orange图片时候:

当分析red图片时候:

当分析yellow图片时候:

(3)结论

当光照变化很大时,我们可以看到:

  • 理想情况下,我们希望使用颜色通道最紧凑/最集中的密度图的颜色空间。
  • RGB的密度图急剧膨胀。这意味着通道值的变化非常大,固定一个阈值是一个大问题。固定较高的范围将检测到与所需颜色相似的颜色(假阳性)和较低的范围将不会检测到不同照明下所需的颜色(假阴性)。
  • 在HSV中,由于只有H分量包含关于绝对颜色的信息。因此,它成为我的颜色空间的第一选择,因为我可以调整一个旋钮(H)来指定颜色,而不是YCrCb (Cr和Cb)和LAB (A和B)中的两个旋钮。
  • 比较YCrCb和LAB的图,发现在LAB的情况下,紧凑程度更高。所以,我的下一个最佳选择是LAB颜色空间。

(4)最终的效果

在最后一节中,我将展示检测蓝色和黄色块的结果,方法是从密度图中提取阈值,并将其应用到各自的颜色空间中,方法与第二节相同。当我们在HSV, YCrCb和LAB颜色空间中工作时,我们不必担心强度分量。我们只需要为颜色分量指定阈值。图中显示了我用来生成结果的值。


在上面的结果中,我直接从密度图中取了值。我们也可以选择取密度图中最密集区域的值,这样可以更严格的控制颜色范围。这将留下一些洞和散乱的像素,可以用侵蚀和膨胀,然后过滤。

3.颜色空间的其他有用应用

  • 直方图均衡化通常是在灰度图像上进行的。但是,您可以通过将RGB图像转换为YCbCr并仅对Y通道进行直方图均衡化来执行彩色图像的均衡化。
  • 两张图像之间通过将图像转换到Lab彩色空间实现彩色变换
  • 许多智能手机的相机应用程序(如谷歌相机或Instagram)的滤镜都利用这些颜色空间转换来创建很酷的效果!

参考目录

https://learnopencv.com/color-spaces-in-opencv-cpp-python/

OpenCV基础(7)OpenCV中的颜色空间相关推荐

  1. opencv如何把一个矩阵不同列分离开_【opencv基础】OpenCV从Mat中提取某些行或列

    前言 Opencv中可以调用函数提取某些连续的行或者列,Mat的rowRange和colRange可以获取某些范围内行或列的指针: 这两个函数返回的是指向原矩阵内部位置的指针,类似于浅拷贝: exam ...

  2. [转载] 使用Python+OpenCV实现在视频中某对象后添加图像

    参考链接: Python Opencv 基础3 : resize 调整图像大小 概述 在运动物体后面添加图像是一个典型的计算机视觉项目了解如何使用传统的计算机视觉技术在视频中添加logo 介绍 我的一 ...

  3. OpenCV基础入门【C++及python语言】

    OpenCV基础入门[C++语言] OpenCV-Python 中文教程 OpenCV官方教程中文版(For Python) OpenCV2-Python-Tutorials 部分文件参考: http ...

  4. Python计算机视觉编程第十章——OpenCV基础知识

    Python计算机视觉编程 (一)OpenCV 的 Python 接口 (二)OpenCV 基础知识 2.1 读取和写入图像 2.2 颜色空间 2.3 显示图像及结果 (三)处理视频 3.1 视频输入 ...

  5. OpenCV基础入门系列基本操作——贰

    系列博文第二篇,关于OpenCV4的一些基本操作和使用. 博文主要以实例展示不同的函数使用方法. OpenCV基础入门系列基本操作--壹 前言 下述为本博文需要用到的各类头文件以及全局变量等 读者可根 ...

  6. PaddlePaddle领航团 OpenCV基础知识点总结

    PaddlePaddle领航团 OpenCV基础知识点总结 1.OpenCV基础 加载图片,显示图片,保存图片 OpenCV函数:cv2.imread(), cv2.imshow(), cv2.imw ...

  7. 5.OpenCV基础

    OpenCV入门 图像 图像是什么 模拟图像和数字图像 数字图像的表示 图像的分类 OpenCV简介 OpenCV-Python OpenCV部署方法 pip install opencv-pytho ...

  8. python open-cv 基础知识总结(三)

    上一章:python open-cv 基础知识总结(二) 1. 轮廓中心计算 本教程的目标是 (1)  检测图像中每个形状的轮廓, (2)  计算轮廓的中心 -也称为区域的  质心 . 为了实现这些目 ...

  9. OpenCV基础之边缘检测与轮廓描绘

    文章目录 OpenCv基础之边缘检测与轮廓描绘 Canny边缘检测 图像轮廓 绘制轮廓 OpenCv基础之边缘检测与轮廓描绘 边缘检测:主要是通过一些手段检测数字图像中明暗变化剧烈(即梯度变化比较大) ...

最新文章

  1. Facebook工程师教你什么是随机森林,就算零基础也可以看懂 | 干货
  2. 腾讯员工人均年薪84.7万,马化腾:员工心理健康最重要
  3. Get data from file(xxx.png) failed!
  4. 技术动态 | eBay开源分布式知识图谱存储Beam,支持类SPARQL查询
  5. C++学习之路 | PTA乙级—— 1087 有多少不同的值 (20 分)(精简)
  6. oracle升级后出现 ora-02055,分布式更新失败 0ra-02055错误 请各位指点(在线等待)...
  7. 万字图解Java多线程,不信你学不会!
  8. Oracle主要概念汇总
  9. 微软ASP.NET官方网站MVC教程实际操作中的部分问题
  10. apache启动失败查看错误信息
  11. JavaSpring菜鸟教程,附Java面经
  12. DB2 查看表空间使用率
  13. 深入浅出PID控制算法(二)——PID算法离散化和增量式PID算法原理及Matlab实现
  14. Mac 安装Nessus
  15. 东芝打印机共享怎么设置_win7系统东芝STUDLO2303A打印机怎么共享网络
  16. 自己做的SIP软电话
  17. Unity中实现3D拾取功能及其原理
  18. 小程序开发用什么编程语言_微信小程序开发教程是什么?费用多少?
  19. cad渐开线齿轮轮廓绘制_CAD画齿轮的方法
  20. jmeter 聚合报告里面是什么意思

热门文章

  1. 为什么onenote一直在加载_OneNote: 沉睡于电脑中的宝藏笔记软件,高效管理你的学习生活...
  2. web服务 面试可能会问的问题
  3. 【ASM】字节码操作 ClassWriter COMPUTE_FRAMES 的作用 与 visitMaxs 的关系
  4. php url 减号,PHP编码转换减号(连接符)无法转换问题
  5. Station M2极客主机
  6. 渗透前期学习资源分享
  7. mysql多次登录失败控制,Mysql登录失败多次锁定配置
  8. [异常检测]Learning Regularity in Skeleton Trajectories for Anomaly Detection in Videos
  9. P1551 亲戚(并查集)
  10. 亲戚关系(并查集(YYOJ