OpenCV實現(xiàn)鼠標框選并顯示框選區(qū)域
本文實例為大家分享了OpenCV實現(xiàn)鼠標框選并顯示框選區(qū)域的具體代碼,供大家參考,具體內(nèi)容如下
cvSetImageROI函數(shù)(基于給定的矩形設置圖像的ROI(感興趣區(qū)域,region of interesting))
void cvSetImageROI(IplImage* image,CvRect rect)
參數(shù):
image 圖像頭,待處理圖像
rect ROI 感興趣區(qū)域矩形
cvResetImageROI函數(shù)(釋放基于給定的矩形設置圖像的ROI(感興趣區(qū)域,region of interesting))
void cvResetImageROI(IplImage* image)
參數(shù):
image 圖像頭,待處理圖像
cvcop函數(shù)(拷貝一個數(shù)組給另一個數(shù)組)
在使用這個函數(shù)之前,你必須用cvCreateImage()一類的函數(shù)先開一段內(nèi)存,然后傳遞給dst。cvCopy會把src中的數(shù)據(jù)復制到dst的內(nèi)存中。
copy只會復制ROI區(qū)域,相當于函數(shù)cvCopy從輸入數(shù)組中復制選定的成分到輸出數(shù)組。
void cvCopy( const CvArr* src, CvArr* dst, const CvArr* mask=NULL );
參數(shù):
src  輸入數(shù)組
dst  輸出數(shù)組
mask  操作掩碼是8比特單通道的數(shù)組,它指定了輸出數(shù)組中被改變的元素 
cvCloneImage函數(shù)(復制圖像數(shù)據(jù))
在使用函數(shù)之前,不用開辟內(nèi)存。該函數(shù)會自己開一段內(nèi)存,然后復制好image里面的數(shù)據(jù),然后把這段內(nèi)存中的數(shù)據(jù)返回給你。
clone是把所有的都復制過來,也就是說不論你是否設置Roi,Coi等影響copy的參數(shù),clone都會原封不動的克隆過來。
IplImage* cvCloneImage( const IplImage* image );
參數(shù):image 輸入源圖像數(shù)據(jù)
返回值:IplImage*  輸出圖像指針
注意:使用cvCloneImage()容易造成內(nèi)存泄露,所以慎用。
cvCloneImage()每次使用時編譯器會分配新的內(nèi)存空間,不會覆蓋以前的內(nèi)容,所以如果在循環(huán)中使用內(nèi)存會迅速減小,每次用完都需要用cvRelease來釋放。
解決方法是使用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) )//鼠標左鍵按下時
 { 
 sprintf(temp,"(%d,%d)",x,y);//格式化字符串
 pre_pt = cvPoint(x,y);//獲取當前點坐標值
 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);//這句有沒有,就是單目標和多目標的問題
 }
 else if( (event == CV_EVENT_MOUSEMOVE) && (flags & CV_EVENT_LBUTTONDOWN))
 {//鼠標移動并且鼠標左鍵按下
 sprintf(temp,"(%d,%d)",x,y);//格式化字符串
 cur_pt = cvPoint(x,y);//獲取當前點坐標值 
 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復制到臨時圖像tmp上,用于實時顯示
 }
 else if( event == CV_EVENT_LBUTTONUP )
 {//鼠標左鍵彈起
 sprintf(temp,"(%d,%d)",x,y);//字體格式化
 cur_pt = cvPoint(x,y);//獲取當前點坐標值 
 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); //兩點橫坐標差 
    int height=abs(pre_pt.y-cur_pt.y); //兩點縱坐標差 
    if(width==0 || height==0) 
    { //兩者中有一個為零時銷毀窗口
      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);//設置圖像的感興趣區(qū)域 
    cvCopy(org,tmp1); //將感興趣區(qū)域復制到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);//復制圖像到臨時圖像上 
 org=cvCloneImage(src);//保存原始圖像 
 cvNamedWindow("src",1);//新建窗口
 cvSetMouseCallback( "src", on_mouse, 0 );//注冊鼠標響應回調(diào)函數(shù)
 
 cvShowImage("src",src);//顯示圖像
 cvWaitKey(0);//等待按鍵按下 
 cvDestroyAllWindows();//銷毀所有窗口
 cvReleaseImage(&src);//釋放圖像
 cvReleaseImage(&tmp);//釋放圖像
 return 0;
}
效果圖:
以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持我們。
上一篇:Qt股票組件之自選股列表拖拽、右鍵常用菜單功能的實現(xiàn)
欄 目:C語言
本文標題:OpenCV實現(xiàn)鼠標框選并顯示框選區(qū)域
本文地址:http://www.jygsgssxh.com/a1/Cyuyan/234.html
您可能感興趣的文章
- 01-10數(shù)據(jù)結(jié)構(gòu)課程設計-用棧實現(xiàn)表達式求值的方法詳解
 - 01-10使用OpenGL實現(xiàn)3D立體顯示的程序代碼
 - 01-10求斐波那契(Fibonacci)數(shù)列通項的七種實現(xiàn)方法
 - 01-10C語言 解決不用+、-、&#215;、&#247;數(shù)字運算符做加法
 - 01-10使用C++實現(xiàn)全排列算法的方法詳解
 - 01-10用C++實現(xiàn)DBSCAN聚類算法
 - 01-10深入全排列算法及其實現(xiàn)方法
 - 01-10全排列算法的非遞歸實現(xiàn)與遞歸實現(xiàn)的方法(C++)
 - 01-10用C語言實現(xiàn)單鏈表的各種操作(一)
 - 01-10用C語言實現(xiàn)單鏈表的各種操作(二)
 


閱讀排行
本欄相關(guān)
- 04-02c語言函數(shù)調(diào)用后清空內(nèi)存 c語言調(diào)用
 - 04-02func函數(shù)+在C語言 func函數(shù)在c語言中
 - 04-02c語言的正則匹配函數(shù) c語言正則表達
 - 04-02c語言用函數(shù)寫分段 用c語言表示分段
 - 04-02c語言中對數(shù)函數(shù)的表達式 c語言中對
 - 04-02c語言編寫函數(shù)冒泡排序 c語言冒泡排
 - 04-02c語言沒有round函數(shù) round c語言
 - 04-02c語言分段函數(shù)怎么求 用c語言求分段
 - 04-02C語言中怎么打出三角函數(shù) c語言中怎
 - 04-02c語言調(diào)用函數(shù)求fibo C語言調(diào)用函數(shù)求
 
隨機閱讀
- 08-05dedecms(織夢)副欄目數(shù)量限制代碼修改
 - 01-10使用C語言求解撲克牌的順子及n個骰子
 - 01-10C#中split用法實例總結(jié)
 - 04-02jquery與jsp,用jquery
 - 01-10SublimeText編譯C開發(fā)環(huán)境設置
 - 08-05DEDE織夢data目錄下的sessions文件夾有什
 - 01-10delphi制作wav文件的方法
 - 01-11ajax實現(xiàn)頁面的局部加載
 - 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
 - 08-05織夢dedecms什么時候用欄目交叉功能?
 


