OpenCV實(shí)現(xiàn)鼠標(biāo)框選并顯示框選區(qū)域
本文實(shí)例為大家分享了OpenCV實(shí)現(xiàn)鼠標(biāo)框選并顯示框選區(qū)域的具體代碼,供大家參考,具體內(nèi)容如下
cvSetImageROI函數(shù)(基于給定的矩形設(shè)置圖像的ROI(感興趣區(qū)域,region of interesting))
void cvSetImageROI(IplImage* image,CvRect rect)
參數(shù):
image 圖像頭,待處理圖像
rect ROI 感興趣區(qū)域矩形
cvResetImageROI函數(shù)(釋放基于給定的矩形設(shè)置圖像的ROI(感興趣區(qū)域,region of interesting))
void cvResetImageROI(IplImage* image)
參數(shù):
image 圖像頭,待處理圖像
cvcop函數(shù)(拷貝一個(gè)數(shù)組給另一個(gè)數(shù)組)
在使用這個(gè)函數(shù)之前,你必須用cvCreateImage()一類的函數(shù)先開一段內(nèi)存,然后傳遞給dst。cvCopy會(huì)把src中的數(shù)據(jù)復(fù)制到dst的內(nèi)存中。
copy只會(huì)復(fù)制ROI區(qū)域,相當(dāng)于函數(shù)cvCopy從輸入數(shù)組中復(fù)制選定的成分到輸出數(shù)組。
void cvCopy( const CvArr* src, CvArr* dst, const CvArr* mask=NULL );
參數(shù):
src 輸入數(shù)組
dst 輸出數(shù)組
mask 操作掩碼是8比特單通道的數(shù)組,它指定了輸出數(shù)組中被改變的元素
cvCloneImage函數(shù)(復(fù)制圖像數(shù)據(jù))
在使用函數(shù)之前,不用開辟內(nèi)存。該函數(shù)會(huì)自己開一段內(nèi)存,然后復(fù)制好image里面的數(shù)據(jù),然后把這段內(nèi)存中的數(shù)據(jù)返回給你。
clone是把所有的都復(fù)制過來(lái),也就是說(shuō)不論你是否設(shè)置Roi,Coi等影響copy的參數(shù),clone都會(huì)原封不動(dòng)的克隆過來(lái)。
IplImage* cvCloneImage( const IplImage* image );
參數(shù):image 輸入源圖像數(shù)據(jù)
返回值:IplImage* 輸出圖像指針
注意:使用cvCloneImage()容易造成內(nèi)存泄露,所以慎用。
cvCloneImage()每次使用時(shí)編譯器會(huì)分配新的內(nèi)存空間,不會(huì)覆蓋以前的內(nèi)容,所以如果在循環(huán)中使用內(nèi)存會(huì)迅速減小,每次用完都需要用cvRelease來(lái)釋放。
解決方法是使用cvCopy函數(shù)代替。
源代碼:
#include <cv.h>
#include <highgui.h>
#include <stdio.h>
IplImage* src = 0;
IplImage* tmp = 0;
IplImage* tmp1 = 0;
IplImage* org = 0;
void on_mouse( int event, int x, int y, int flags, void* ustc)
{
static CvPoint pre_pt = {-1,-1};
static CvPoint cur_pt = {-1,-1};
CvFont font;
cvInitFont(&font, CV_FONT_HERSHEY_SIMPLEX, 0.5, 0.5, 0, 1, CV_AA);//初始化字體
char temp[16];
if( (event == CV_EVENT_LBUTTONDOWN)&&(flags) )//鼠標(biāo)左鍵按下時(shí)
{
sprintf(temp,"(%d,%d)",x,y);//格式化字符串
pre_pt = cvPoint(x,y);//獲取當(dāng)前點(diǎn)坐標(biāo)值
cvPutText(src,temp, pre_pt, &font, cvScalar(0,0, 0, 255));//在圖像是打印字符
cvCircle( src, pre_pt, 2,cvScalar(255,0,0,0) ,CV_FILLED, CV_AA, 0 );//在圖像上畫圓
cvShowImage( "src", src );
//cvCopy(src,tmp);//這句有沒有,就是單目標(biāo)和多目標(biāo)的問題
}
else if( (event == CV_EVENT_MOUSEMOVE) && (flags & CV_EVENT_LBUTTONDOWN))
{//鼠標(biāo)移動(dòng)并且鼠標(biāo)左鍵按下
sprintf(temp,"(%d,%d)",x,y);//格式化字符串
cur_pt = cvPoint(x,y);//獲取當(dāng)前點(diǎn)坐標(biāo)值
cvPutText(src,temp, cur_pt, &font, cvScalar(0,0, 0, 255));//在圖像是打印字符
cvRectangle(src, pre_pt, cur_pt, cvScalar(0,255,0,0), 2, 8, 0 );//在圖像上畫矩形
cvShowImage( "src", src );
cvCopy(tmp,src);//將img復(fù)制到臨時(shí)圖像tmp上,用于實(shí)時(shí)顯示
}
else if( event == CV_EVENT_LBUTTONUP )
{//鼠標(biāo)左鍵彈起
sprintf(temp,"(%d,%d)",x,y);//字體格式化
cur_pt = cvPoint(x,y);//獲取當(dāng)前點(diǎn)坐標(biāo)值
cvPutText(src,temp, cur_pt, &font, cvScalar(0,0, 0, 255));//在圖像是打印字符
cvCircle( src, cur_pt, 2,cvScalar(255,0,0,0) ,CV_FILLED, CV_AA, 0 );//在圖像上畫圓
cvRectangle( src, pre_pt, cur_pt, cvScalar(0,255,0,0), 2, 8, 0 );//在圖像上畫矩形
cvShowImage( "src", src );
/******************************************************************/
int width=abs(pre_pt.x-cur_pt.x); //兩點(diǎn)橫坐標(biāo)差
int height=abs(pre_pt.y-cur_pt.y); //兩點(diǎn)縱坐標(biāo)差
if(width==0 || height==0)
{ //兩者中有一個(gè)為零時(shí)銷毀窗口
cvDestroyWindow("dst");
return;
}
tmp1 = cvCreateImage(cvSize(width,height),org->depth,org->nChannels);
CvRect rect;
if(pre_pt.x<cur_pt.x && pre_pt.y<cur_pt.y)
{
rect=cvRect(pre_pt.x,pre_pt.y,width,height);
}
else if(pre_pt.x>cur_pt.x && pre_pt.y<cur_pt.y)
{
rect=cvRect(cur_pt.x,pre_pt.y,width,height);
}
else if(pre_pt.x>cur_pt.x && pre_pt.y>cur_pt.y)
{
rect=cvRect(cur_pt.x,cur_pt.y,width,height);
}
else if(pre_pt.x<cur_pt.x && pre_pt.y>cur_pt.y)
{
rect=cvRect(pre_pt.x,cur_pt.y,width,height);
}
cvSetImageROI(org,rect);//設(shè)置圖像的感興趣區(qū)域
cvCopy(org,tmp1); //將感興趣區(qū)域復(fù)制到tmp1
cvResetImageROI(org);//釋放圖像的感興趣區(qū)域
cvDestroyWindow("dst");//銷毀上次的顯示圖像
cvNamedWindow("dst",1);//新建窗口
cvShowImage("dst",tmp1); //顯示感興趣的圖像
cvSaveImage("dst.jpg",tmp1); //保存感興趣圖像
/******************************************************************/
}
}
int main()
{
src=cvLoadImage("lena.jpg",1);//讀入圖像
tmp=cvCloneImage(src);//復(fù)制圖像到臨時(shí)圖像上
org=cvCloneImage(src);//保存原始圖像
cvNamedWindow("src",1);//新建窗口
cvSetMouseCallback( "src", on_mouse, 0 );//注冊(cè)鼠標(biāo)響應(yīng)回調(diào)函數(shù)
cvShowImage("src",src);//顯示圖像
cvWaitKey(0);//等待按鍵按下
cvDestroyAllWindows();//銷毀所有窗口
cvReleaseImage(&src);//釋放圖像
cvReleaseImage(&tmp);//釋放圖像
return 0;
}
效果圖:

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
C++內(nèi)核對(duì)象封裝單實(shí)例啟動(dòng)程序的類
這篇文章主要介紹了利用C++內(nèi)核對(duì)象封裝的類,程序只能運(yùn)行單個(gè)實(shí)例,可防止多次啟動(dòng),大家參考使用吧2013-11-11
C語(yǔ)言實(shí)例梳理講解常用關(guān)鍵字的用法
關(guān)鍵字是C語(yǔ)言非常重要的一部分,熟練的掌握和使用關(guān)鍵字有助于我們更加熟悉了解C語(yǔ)言,同時(shí)C語(yǔ)言的關(guān)鍵字也是面試筆試中??嫉膬?nèi)容。C語(yǔ)言的關(guān)鍵字共有32個(gè),但并不是每個(gè)關(guān)鍵字都有坑,本篇文章將通過理論聯(lián)系實(shí)際的方式為大家講解C語(yǔ)言中易混易錯(cuò)以及??嫉囊恍╆P(guān)鍵字2022-05-05
C++實(shí)現(xiàn)簡(jiǎn)易的通訊錄管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了C++實(shí)現(xiàn)簡(jiǎn)易的通訊錄管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-06-06
基于OpenCV和C++ 實(shí)現(xiàn)圖片旋轉(zhuǎn)
這篇文章主要介紹了基于OpenCV和C++ 實(shí)現(xiàn)圖片旋轉(zhuǎn),幫助大家更好的利用c++處理圖片,感興趣的朋友可以了解下2020-12-12
CRC校驗(yàn)原理及其C語(yǔ)言實(shí)現(xiàn)詳解
循環(huán)冗余校驗(yàn)(Cyclic?Redundancy?Check,?CRC)是一種根據(jù)網(wǎng)絡(luò)數(shù)據(jù)包或計(jì)算機(jī)文件等數(shù)據(jù)產(chǎn)生簡(jiǎn)短固定位數(shù)校驗(yàn)碼的一種信道編碼技術(shù)。本文主要介紹了CRC校驗(yàn)原理及其C語(yǔ)言實(shí)現(xiàn),感興趣的可以了解一下2023-03-03
C++標(biāo)準(zhǔn)之(ravalue reference) 右值引用介紹
臨時(shí)對(duì)象的產(chǎn)生和拷貝所帶來(lái)的效率折損,一直是C++所為人詬病的問題,下面簡(jiǎn)單地介紹一下Copy Elision、RVO,對(duì)此不感興趣的可以直接跳過2012-11-11

