https://blog.csdn.net/ZYTTAE/article/details/42234989
由於算法的需要,需要把彩色圖像的R、G、B值分離出來,OpenCV中正好有split() 和 merge() 函數可以實現。
一、對單獨彩色圖片的RGB通道分離:
#include <iostream> #include "cv.h" #include "highgui.h" using namespace std; using namespace cv; int main(int argc,char* argv[]) { Mat img = imread("lena.jpg"/*,CV_LOAD_IMAGE_COLOR*/); Mat channel[3]; split(img,channel); imshow("original",img); imshow("B",channel[0]); imshow("G",channel[1]); imshow("R",channel[2]); //set blue channel to 0 channel[0] = Mat::zeros(img.rows,img.cols,CV_8UC1); //merge red and green channels merge(channel,3,img); imshow("R_G_merge",img); waitKey(0); return 1; }
二、對攝像頭攝入視頻幀的RGB彩色通道分離
int main(int argc,char* argv[]) { VideoCapture cap; cap.open(0); if(!cap.isOpened()) { exit(0); } cap.set(CV_CAP_PROP_FRAME_WIDTH,250); cap.set(CV_CAP_PROP_FRAME_HEIGHT,250); cout << "Frame Width: " << cap.get(CV_CAP_PROP_FRAME_WIDTH) << endl; cout << "Frame Height: " << cap.get(CV_CAP_PROP_FRAME_HEIGHT) << endl; Mat frame; vector<Mat> rgb; cap >> frame; //rgb.push_back( Mat(frame.rows, frame.cols, CV_8UC1)); //rgb.push_back( Mat(frame.rows, frame.cols, CV_8UC1)); //rgb.push_back( Mat(frame.rows, frame.cols, CV_8UC1)); //rgb.push_back( Mat(frame.rows, frame.cols, CV_8UC1)); namedWindow("original", 1); namedWindow("red", 1); namedWindow("green", 1); namedWindow("blue", 1); for(;;) { cap >> frame; imshow("original", frame); split(frame, rgb); imshow("red", rgb.at(2)); imshow("green", rgb.at(1)); imshow("blue", rgb.at(0)); if(waitKey(30) >= 0) break; } waitKey(0); return 1; }
1.split()函數
此函數的作用是將一個圖像通道進行分離。
首先看一下split()函數定義:
void split(const Mat& m, vector<Mat>& mv );
參數說明:
第一個參數,const Mat&類型的src,填我們需要進行分離的圖像;
第二個參數,vector<Mat>類型的mv,填函數的輸出數組或者輸出的vector容器,即分離后的圖像;
2.merge()函數
merge()函數的功能是split()函數的逆向操作,將多個數組合並成一個多通道的數組。
首先看一下merge()函數定義:
void merge(const vector<Mat>& mv, OutputArray dst );
參數說明:
第一個參數,const <Mat>類型的mv,填需要被合並的vector容器的陣列,這個mv參數中所有的矩陣必須有着一樣的尺寸和深度;說白了就是前面被split()函數分離后的圖像通道。
第二個參數,保存為合並后的圖像;
#include <opencv2/core/core.hpp> #include <opencv2/highgui/highgui.hpp> #include <opencv2/imgproc/imgproc.hpp> #include <iostream> using namespace cv; using namespace std; int main() { //【0】定義相關變量 Mat srcImage, newImage; //源圖像、通道合並后的圖像 Mat srcImage_B, srcImage_G, srcImage_R; //R、G、B各個通道 Mat image_H, image_S, image_V; //H、S、V各個通道 vector<Mat> channels_BGR; //vector<Mat>: 可以理解為存放Mat類型的容器(數組) vector<Mat> channels_HSV; //【1】讀取原始圖像並檢查圖像是否讀取成功 srcImage = imread("D:\\OutPutResult\\ImageTest\\adog.jpg"); //請修改為自己的圖像路徑 if (srcImage.empty()) { cout << "讀取圖像有誤,請重新輸入正確路徑!\n"; return -1; } imshow("srcImage源圖像", srcImage); //在窗口顯示源圖像 //【2】對加載的原圖像進行通道分離,即把一個3通道圖像轉換成為3個單通道圖像 split(srcImage, channels_BGR); //0通道為B分量,1通道為G分量,2通道為R分量。因為:RGB色彩空間在opencv中默認通道順序為BGR!!! srcImage_B = channels_BGR.at(0); srcImage_G = channels_BGR.at(1); srcImage_R = channels_BGR.at(2); imshow("srcImage_B通道", srcImage_B); //分別顯示R,G,B各個通道圖像 imshow("srcImage_G通道", srcImage_G); imshow("srcImage_R通道", srcImage_R); //【3】將BGR顏色空間轉換為HSV顏色空間 Mat image_hsv; cvtColor(srcImage, image_hsv, CV_BGR2HSV); imshow("HSV顏色空間圖像", image_hsv); //【4】對加載的HSV圖像進行通道分離 split(image_hsv, channels_HSV); //0通道為H分量,1通道為S分量,2通道為V分量 image_H = channels_HSV.at(0); image_S = channels_HSV.at(1); image_V = channels_HSV.at(2); imshow("image_H通道", image_H);//分別顯示H,S,V各個通道圖像 imshow("image_S通道", image_S); imshow("image_V通道", image_V); //【5】將3個單通道重新合並成一個三通道圖像 merge(channels_HSV, newImage); imshow("將H,S,V通道合並后的圖像", newImage); //【6】保持等待狀態 waitKey(0); return 0; }
5.程序說明 看到這里,可能有人會問為什么分離出的通道都是黑白灰,而不是紅綠藍? 原因是分離后為單通道,相當於分離通道的同時把其他兩個通道填充了相同的數值。比如紅色通道,分離出紅色通道的同時,綠色和藍色被填充為和紅色相同的數值,這樣一來就只有黑白灰了。那么紅色體現在哪呢?可以進行觀察,會發現原圖中顏色越接近紅色的地方在紅色通道越接近白色。 ====================分割線=============== 此程序共顯示9個窗口。 先將RGB圖像通道分離,分別顯示R、G、B、單個通道; 后將RGB顏色空間轉為HSV空間,將HSV圖像通道分離,分別顯示H、S、V、單個通道; 最后將H、S、V、單個通道重新合並為3通道圖像; =========================END======================