C++使用opencv調(diào)用級(jí)聯(lián)分類器來(lái)識(shí)別目標(biāo)物體的詳細(xì)流程
使用編譯器:Qt Creator 4.2.1
前言:
相較于幀差法捕捉目標(biāo)物體識(shí)別,級(jí)聯(lián)分類器識(shí)別目標(biāo)物體更加具有針對(duì)性,使用前者只要是動(dòng)的物體都會(huì)被捕捉識(shí)別到,畫(huà)面里有一點(diǎn)風(fēng)吹草動(dòng),都會(huì)被捕捉識(shí)別下來(lái),如果我想識(shí)別具體的人或者物,都無(wú)法做到精準(zhǔn)的目標(biāo)識(shí)別,所以有了級(jí)聯(lián)分類器識(shí)別(即模型識(shí)別),會(huì)按照訓(xùn)練好的級(jí)聯(lián)分類器(模型)來(lái)進(jìn)行目標(biāo)識(shí)別
流程講解:
1.創(chuàng)建一個(gè)級(jí)聯(lián)分類器對(duì)象
創(chuàng)建一個(gè)級(jí)聯(lián)分類器對(duì)象,并讀取已經(jīng)已經(jīng)訓(xùn)練好的模型
CascadeClassifier cascade;//級(jí)聯(lián)分類器(模型)
cascade.load("D:/Qt_Opencv_Project/cars.xml");//讀取級(jí)聯(lián)分類器2.創(chuàng)建一個(gè)視頻流
讀取一個(gè)要識(shí)別的視頻路徑
Mat frame;
VideoCapture cap("D:/VideoTraining/carMove.mp4");
while (cap.read(frame))
{
imshow("video",frame);//將讀到的幀顯示出來(lái)
datectCarDaw(frame,cascade,2);//將讀到的幀傳入函數(shù)用作識(shí)別
waitKey(25);//延時(shí)25ms,避免播放過(guò)快
}3.將傳入的視頻幀轉(zhuǎn)換為灰度圖
轉(zhuǎn)換為灰度圖,色彩通道縮小一半,減少圖片數(shù)據(jù)計(jì)算量,提升計(jì)算速度
Mat gray;
//轉(zhuǎn)換為灰度圖,色彩通道縮小一半
cvtColor(frame,gray,CV_RGB2GRAY);效果如下:

4.將多余的空通道進(jìn)行壓縮
原通道為RGB三通道圖片數(shù)據(jù),轉(zhuǎn)換為灰度圖后,變?yōu)閱瓮ǖ罃?shù)據(jù),多余的通道可以壓縮掉,可以看到圖片減小三分之二
//再將灰度圖縮小一半
Mat smalling(cvRound(frame.rows/scale),cvRound(frame.cols/scale),CV_8UC1);
resize(gray,smalling,smalling.size(),0,0,INTER_LINEAR);效果如下:

可以發(fā)現(xiàn),圖一下子小了很多,大大降低了計(jì)算量
5.直方圖均衡化
將縮小一半的灰度圖進(jìn)行均衡化,使其更加黑白分明,增強(qiáng)局部的對(duì)比度,便于計(jì)算機(jī)識(shí)別
//直方圖均衡化:將縮小一半的灰度圖進(jìn)行均值化 使其更加黑白分明
equalizeHist(smalling,smalling);效果如下:

降低了漸進(jìn)效果的像素?cái)?shù)目,更加的黑白分明
6.調(diào)用級(jí)聯(lián)分類器,并將內(nèi)容框出,然后將此幀顯示出來(lái)
調(diào)用讀取已經(jīng)訓(xùn)練好的模型,調(diào)用detectMultiScale函數(shù)器參數(shù)含義為:
待檢測(cè)的圖片幀(此處為均衡化后的圖片幀)
被檢測(cè)物體的矩形向量容器(調(diào)用前聲明了一個(gè)向量容器cars)
每次搜索減小的圖像比例(每次縮小1.05倍,掃描的細(xì)致一點(diǎn))
檢測(cè)目標(biāo)周?chē)噜従匦蔚淖钚€(gè)數(shù)(此處設(shè)為5個(gè),此處視道路情況可以適當(dāng)增加個(gè)數(shù))
類型(掃描類型)
目標(biāo)區(qū)域可能的最小尺寸(此處為25*25,太小了會(huì)識(shí)別不到,太大了可能幾輛車(chē)貼的很近可能只能識(shí)別出一輛)
//調(diào)用級(jí)聯(lián)分類器進(jìn)行模型匹配并框出內(nèi)容
vector<Rect>cars;
//此函數(shù)參數(shù)說(shuō)明: 待檢測(cè)的圖片幀 被檢測(cè)物體的矩形向量容器 每次搜索減小的圖像比例
//檢測(cè)目標(biāo)周?chē)噜従匦蔚淖钚€(gè)數(shù)(此處設(shè)為2個(gè)) 類型 目標(biāo)區(qū)域的大小尺寸
cascade.detectMultiScale(smalling,cars,1.05,5,0|CV_HAAR_SCALE_IMAGE,Size(25,25));
vector<Rect>::const_iterator iter;
//繪制方塊,標(biāo)記目標(biāo),注意,標(biāo)記要畫(huà)在原幀上,要講方框的大小和幀坐標(biāo)擴(kuò)大,因?yàn)槭歉鶕?jù)灰度圖識(shí)別的,灰度圖被縮小了
for(iter=cars.begin();iter!=cars.end();iter++)
{
rectangle(frame,
cvPoint(cvRound(iter->x*scale),cvRound(iter->y*scale)),
cvPoint(cvRound((iter->x+iter->width)*scale),cvRound((iter->y+iter->height)*scale)),
Scalar(0,255,0),2,8
);
}
imshow("frame",frame);效果如下:

循環(huán)顯示每一幀,最終成果展示:
博主所使用的這個(gè)模型訓(xùn)練的樣本較為有限,主要訓(xùn)練的是普通轎車(chē)的車(chē)頭,車(chē)尾訓(xùn)練較少,貨車(chē)等其他車(chē)型沒(méi)有做特別的訓(xùn)練,所以只能保證識(shí)別轎車(chē)的車(chē)頭

可以根據(jù)實(shí)際車(chē)流量和車(chē)輛距離攝像頭的距離來(lái)調(diào)整最小目標(biāo)大小和最大矩形框個(gè)數(shù)
代碼:
#include <iostream>
#include <opencv2/opencv.hpp>
using namespace cv;
using namespace std;
void datectCarDaw(Mat &frame,CascadeClassifier cascade,double scale)
{
Mat gray;
//轉(zhuǎn)換為灰度圖,色彩通道縮小一半
cvtColor(frame,gray,CV_RGB2GRAY);
//imshow("huidu",gray);
//再將灰度圖縮小一半
Mat smalling(cvRound(frame.rows/scale),cvRound(frame.cols/scale),CV_8UC1);
resize(gray,smalling,smalling.size(),0,0,INTER_LINEAR);
//直方圖均衡化:將縮小一半的灰度圖進(jìn)行均值化 使其更加黑白分明
equalizeHist(smalling,smalling);
//imshow("smalling",smalling);
//調(diào)用級(jí)聯(lián)分類器進(jìn)行模型匹配并框出內(nèi)容
vector<Rect>cars;
//此函數(shù)參數(shù)說(shuō)明: 待檢測(cè)的圖片幀 被檢測(cè)物體的矩形向量容器 每次搜索減小的圖像比例 檢測(cè)目標(biāo)周?chē)噜従匦蔚淖钚€(gè)數(shù)(此處設(shè)為2個(gè)) 類型 目標(biāo)區(qū)域的大小尺寸
cascade.detectMultiScale(smalling,cars,1.05,5,0|CV_HAAR_SCALE_IMAGE,Size(25,25));
vector<Rect>::const_iterator iter;
//繪制方塊,標(biāo)記目標(biāo),注意,標(biāo)記要畫(huà)在原幀上,要講方框的大小和幀坐標(biāo)擴(kuò)大,因?yàn)槭歉鶕?jù)灰度圖識(shí)別的,灰度圖被縮小了
for(iter=cars.begin();iter!=cars.end();iter++)
{
rectangle(frame,
cvPoint(cvRound(iter->x*scale),cvRound(iter->y*scale)),
cvPoint(cvRound((iter->x+iter->width)*scale),cvRound((iter->y+iter->height)*scale)),
Scalar(0,255,0),2,8
);
}
imshow("frame",frame);
}
int main(int argc, char *argv[])
{
CascadeClassifier cascade;//級(jí)聯(lián)分類器(模型)
cascade.load("D:/Qt_Opencv_Project/cars.xml");//讀取級(jí)聯(lián)分類器
//cascade.load("D:/Qt_Opencv_Project/face.xml");
Mat frame;
VideoCapture cap("D:/VideoTraining/carMove.mp4");
//VideoCapture cap(0);
while (cap.read(frame))
{
imshow("video",frame);//將讀到的幀顯示出來(lái)
datectCarDaw(frame,cascade,2);//將讀到的幀傳入函數(shù)用作識(shí)別
waitKey(25);//延時(shí)25ms,避免播放過(guò)快
}
return 0;
}有興趣也可以看看無(wú)需使用模型但是只能捕捉動(dòng)態(tài)目標(biāo)的幀差法識(shí)別:
C++調(diào)用opencv完成運(yùn)動(dòng)目標(biāo)捕捉
Qt Creator配置使用opencv:
總結(jié)
到此這篇關(guān)于C++使用opencv調(diào)用級(jí)聯(lián)分類器來(lái)識(shí)別目標(biāo)物體的文章就介紹到這了,更多相關(guān)C++ opencv識(shí)別目標(biāo)物體內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Windows程序內(nèi)部運(yùn)行機(jī)制實(shí)例詳解
這篇文章主要介紹了Windows程序內(nèi)部運(yùn)行機(jī)制實(shí)例詳解,對(duì)于學(xué)習(xí)Windows程序設(shè)計(jì)來(lái)說(shuō)是非常重要的一課,需要的朋友可以參考下2014-08-08
C++二維數(shù)組中數(shù)組元素存儲(chǔ)地址的計(jì)算疑問(wèn)講解
今天小編就為大家分享一篇關(guān)于C++二維數(shù)組中數(shù)組元素存儲(chǔ)地址的計(jì)算疑問(wèn)講解,小編覺(jué)得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來(lái)看看吧2019-02-02
C語(yǔ)言數(shù)據(jù)結(jié)構(gòu)之順序數(shù)組的實(shí)現(xiàn)
這篇文章主要介紹了C語(yǔ)言數(shù)據(jù)結(jié)構(gòu)之順序數(shù)組的實(shí)現(xiàn)的相關(guān)資料,這里提供實(shí)現(xiàn)實(shí)例,希望通過(guò)本文能幫助到大家,需要的朋友可以參考下2017-08-08

