Camshift目标跟踪

Camshift算法是Continuously Adaptive Mean Shift algorithm的简称。它是一个基于MeanSift的改进算法。它首次由Gary R.Bradski等人提出和应用在人脸的跟踪上,并取得了不错的效果。由于它是利用颜色的概率信息进行的跟踪,使得它的运行效率比较高。Camshift算法的过程由下面步骤组成:

(1)确定初始目标及其区域;

(2)计算出目标的色度(Hue)分量的直方图;

(3)利用直方图计算输入图像的反向投影图(后面做进一步的解释);

(4)利用MeanShift算法在反向投影图中迭代收索,直到其收敛或达到最大迭代次数。并保存零次矩;

(5)从第(4)步中获得收索窗口的中心位置和计算出新的窗口大小,以此为参数,进入到下一幀的目标跟踪。(即跳转到第(2)步);

几点说明:

1. 在输入图像进行反向投影图之前在HSV空间内做了一个阀值处理,用以滤掉一些噪声。

2. 反向投影图则是概率分布图,在反向投影图中某一像素点的值指的是这个点符合目标的概率分布的概率是多少,或者直接说其为目标图像像素点的像素点是多少。计算方法为:根据像素点的像素值查目标的直方图,其对应像素值的概率是多少就做为该点在反向投影图中的值。

3. Camshit算法到底是怎样自适应调整窗口的大小的。扩大:Canshift算法在计算窗口大小前,在MeanShift算出的窗口的四个方向上增大了TOLERANCE,即高和宽都增大了2TOLERANCE(此值自己调整设置),这才有可能使得窗口能够变大。缩小:在扩大的窗口内重新计算0阶矩,1阶矩和2阶矩,利用矩的值重新计算高和宽。因此Camshif算法相当于在MeanShift的结果上,再做了一个调整,从而使得跟踪的窗口大小能够随目标的大小变化。

优点:算法的效率比较高,如果能利用多少特征做出来的统计直方图,我估计实验效果会更好。

缺点:(1)只利用颜色统计做的跟踪,在背景有相似颜色时,会出现跟踪错误的情况。(2)不能做多目标跟踪。(3)由于它只在初始位置(而不是从每个像素点)开始迭代,所以有可能在初始位置错了后,收敛的位置还是原位置(即跟丢了后,可能会找不回来)。

问题:论文中有关于窗口大小调整,是根据直方图来迭代求解,不知是怎么回事?在代码中没看到实现。在此向大家请教!

下面是Camshift算法Demo的代码:

代码

 #pragma package
 #endif
 
 #define CV_NO_BACKWARD_COMPATIBILITY
 
 #ifndef _EiC
 #include "cv.h"
 #include "highgui.h"
 #include
 #include
 #endif
 
 IplImage *image = 0, *hsv = 0, *hue = 0, *mask = 0, *backproject = 0, *histimg = 0;
 CvHistogram *hist = 0;
 
 int backproject_mode = 0;
 int select_object = 0;
 int track_object = 0;
 int show_hist = 1;
 CvPoint origin;
 CvRect selection;
 CvRect track_window;
 CvBox2D track_box;
 CvConnectedComp track_comp;
 int hdims = 16;
 float hranges_arr[] = {0,180};
 float* hranges = hranges_arr;
 int vmin = 10, vmax = 256, smin = 30;
 
 void on_mouse( int event, int x, int y, int flags, void* param )
 {
     if( !image )
         return;
 
     if( image->origin )
         y = image->height - y;
 
     if( select_object )//表明还正在框选目标
     {
         selection.x = MIN(x,origin.x);
         selection.y = MIN(y,origin.y);
         selection.width = selection.x + CV_IABS(x - origin.x);
         selection.height = selection.y + CV_IABS(y - origin.y);
 
         //保证数据的有效性
         selection.x = MAX( selection.x, 0 );
         selection.y = MAX( selection.y, 0 );
         selection.width = MIN( selection.width, image->width );
         selection.height = MIN( selection.height, image->height );
         selection.width -= selection.x;
         selection.height -= selection.y;
     }
 
     switch( event )
     {
     case CV_EVENT_LBUTTONDOWN://框选目标
         origin = cvPoint(x,y);
         selection = cvRect(x,y,0,0);
         select_object = 1;
         break;
     case CV_EVENT_LBUTTONUP://框选结束
         select_object = 0;
         if( selection.width > 0 && selection.height > 0 )
             track_object = -1;
         break;
     }
 }
 
 
 CvScalar hsv2rgb( float hue )
 {
     int rgb[3], p, sector;
     static const int sector_data[][3]=
         {{0,2,1}, {1,2,0}, {1,0,2}, {2,0,1}, {2,1,0}, {0,1,2}};
     hue *= 0.033333333333333333333333333333333f;
     sector = cvFloor(hue);
     p = cvRound(255*(hue - sector));
     p ^= sector & 1 ? 255 : 0;
 
     rgb[sector_data[sector][0]] = 255;
     rgb[sector_data[sector][1]] = 0;
     rgb[sector_data[sector][2]] = p;
 
     return cvScalar(rgb[2], rgb[1], rgb[0],0);
 }
 
 int main( int argc, char** argv )
 {
     CvCapture* capture = 0;
 
     if( argc == 1 || (argc == 2 && strlen(argv[1]) == 1 && isdigit(argv[1][0])))
         capture = cvCaptureFromCAM( argc == 2 ? argv[1][0] - '0' : 0 );
     else if( argc == 2 )
         capture = cvCaptureFromAVI( argv[1] );
 
     if( !capture )
     {
         fprintf(stderr,"Could not initialize capturing...\n");
         return -1;
     }
 
     printf( "Hot keys: \n"
         "\tESC - quit the program\n"
         "\tc - stop the tracking\n"
         "\tb - switch to/from backprojection view\n"
         "\th - show/hide object histogram\n"
         "To initialize tracking, select the object with mouse\n" );
 
     cvNamedWindow( "Histogram", 1 );
     cvNamedWindow( "CamShiftDemo", 1 );
     cvSetMouseCallback( "CamShiftDemo", on_mouse, 0 );
     cvCreateTrackbar( "Vmin", "CamShiftDemo", &vmin, 256, 0 );
     cvCreateTrackbar( "Vmax", "CamShiftDemo", &vmax, 256, 0 );
     cvCreateTrackbar( "Smin", "CamShiftDemo", &smin, 256, 0 );
 
     for(;;)
     {
         IplImage* frame = 0;
         int i, bin_w, c;
 
         frame = cvQueryFrame( capture );
         if( !frame )
             break;
 
         if( !image )
         {
            
             image = cvCreateImage( cvGetSize(frame), 8, 3 );
             image->origin = frame->origin;
             hsv = cvCreateImage( cvGetSize(frame), 8, 3 );
             hue = cvCreateImage( cvGetSize(frame), 8, 1 );
             mask = cvCreateImage( cvGetSize(frame), 8, 1 );
             backproject = cvCreateImage( cvGetSize(frame), 8, 1 );
             hist = cvCreateHist(1, &hdims, CV_HIST_ARRAY, &hranges, 1 );
             histimg = cvCreateImage( cvSize(320,200), 8, 3 );
             cvZero( histimg );
         }
 
         cvCopy( frame, image, 0 );
         cvCvtColor( image, hsv, CV_BGR2HSV );
 
         if( track_object )
         {
             int _vmin = vmin, _vmax = vmax;
 
             cvInRangeS( hsv, cvScalar(0,smin,MIN(_vmin,_vmax),0),
                         cvScalar(180,256,MAX(_vmin,_vmax),0), mask ); //去除噪声,在此数据内的值,确定mask为1
             cvSplit( hsv, hue, 0, 0, 0 ); //获得色调分量,并以此来做反向投影图
 
             if( track_object < 0 )
             {
                 float max_val = 0.f;
                 cvSetImageROI( hue, selection );
                 cvSetImageROI( mask, selection );
                 cvCalcHist( &hue, hist, 0, mask );//计算选中部分直方图
                 cvGetMinMaxHistValue( hist, 0, &max_val, 0, 0 );
                 cvConvertScale( hist->bins, hist->bins, max_val ? 255. / max_val : 0., 0 );
                 cvResetImageROI( hue );
                 cvResetImageROI( mask );
                 track_window = selection;
                 track_object = 1;
 
                 cvZero( histimg );
                 bin_w = histimg->width / hdims;
                 for( i = 0; i < hdims; i++ )
                 {
                     int val = cvRound( cvGetReal1D(hist->bins,i)*histimg->height/255 );//获取直方图的中每一项的高
                     CvScalar color = hsv2rgb(i*180.f/hdims);//直方图每一项的颜色是根据项数变化的
                     cvRectangle( histimg, cvPoint(i*bin_w,histimg->height),  //画直方图
                                  cvPoint((i+1)*bin_w,histimg->height - val),
                                  color, -1, 8, 0 );
                 }
             }
 
             cvCalcBackProject( &hue, backproject, hist ); //计算反向投影图backproject
             cvAnd( backproject, mask, backproject, 0 );        //去除上下阀值外的点后的投影图
             cvCamShift( backproject, track_window,        //利用camshift搜索0-255的灰度图像
                         cvTermCriteria( CV_TERMCRIT_EPS | CV_TERMCRIT_ITER, 10, 1 ),
                         &track_comp, &track_box );
             track_window = track_comp.rect;    //获得新的跟踪窗口
 
             if( backproject_mode )
                 cvCvtColor( backproject, image, CV_GRAY2BGR );
        
             if( !image->origin ) //如果为假,需要改变椭圆的角度
                 track_box.angle = -track_box.angle;
             cvEllipseBox( image, track_box, CV_RGB(255,0,0), 3, CV_AA, 0 );//画跟踪椭圆
         }
 
         if( select_object && selection.width > 0 && selection.height > 0 )//在框住的时候反向显示
         {
             cvSetImageROI( image, selection );
             cvXorS( image, cvScalarAll(255), image, 0 );
             cvResetImageROI( image );
         }
 
         cvShowImage( "CamShiftDemo", image );
         cvShowImage( "Histogram", histimg );
 
         c = cvWaitKey(10);
         if( (char) c == 27 )
             break;
         switch( (char) c )
         {
         case 'b':
             backproject_mode ^= 1;
             break;
         case 'c':
             track_object = 0;
             cvZero( histimg );
             break;
         case 'h':
             show_hist ^= 1;
             if( !show_hist )
                 cvDestroyWindow( "Histogram" );
             else
                 cvNamedWindow( "Histogram", 1 );
             break;
         default:
             ;
         }
     }
 
     cvReleaseCapture( &capture );
     cvDestroyWindow("CamShiftDemo");
 
     return 0;
 }
 
 #ifdef _EiC
 main(1,"camshiftdemo.c");
 #endif



你可能感兴趣的:(目标跟踪)