原创 基于opencv的手势识别(转)

2009-2-24 11:45 13739 9 11 分类: 软件与OS
OpenCV模板匹配的方法识别手势(修改版)

找不原出处。


//修改后能运行的程序,但是结果好像不是很好
//我的机子是赛扬600,超频到900在用,内存是256sdram,刚才调的时候居然
//内存不够用了,又很多地方需要改进,如果有时间,我会把改了的重新上传
//还有个问题就是,偶尔会定位到我的脸上,估计是我的脸长的太像手了 : )
//采用模板匹配的方法识别手势

//这次修改了内存消耗猛增的bug,但是仍然有小幅增长,只要测试时间在半个
//小时之内,一般不会出现内存不足的提示。
//修正了原来版本在读取10幅图后显示错误的问题
//测试图片请在《[资料]OpenCV模板匹配的方法识别手势》里面下载

#ifdef _CH_
#define WIN32
#error "The file needs cvaux, which is not wrapped yet. Sorry"
#endif

#ifndef _EiC
#include "cv.h"
#include "cxcore.h"
#include "cvaux.h"
#include "highgui.h"

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <assert.h>
#include <math.h>
#include <float.h>
#include <limits.h>
#include <time.h>
#include <ctype.h>
#endif


static CvMemStorage* storage;
//static double color;
static CvSize imgSize;
static CvSize tmplSize;

static CvSeq* contour;
static CvSeq* biggest;
static CvCapture* capture = 0;
static int isHandOpen = 1;
//static double color;
static CvScalar color;
static IplImage *img;
static IplImage *tmpImg;
static IplImage *conv;
static IplImage *H;
static IplImage *S;
static IplImage *V;
static IplImage *tmpH1;
static IplImage *tmpS1;
static IplImage *tmpH2;
static IplImage *tmpS2;
static IplImage *tmpH3;
static IplImage *tmpS3;
static IplImage *openHandTmpl, *closedHandTmpl;
static IplImage *openGrayHandTmpl, *closedGrayHandTmpl;
static IplImage *openscaledTmpl, *closedscaledTmpl;
static IplImage *openMatchResult, *closedMatchResult;

FILE *output;

void handDetect( IplImage *img);

static int numImg = 10;

char
*names[] = { "./images/test01.jpg", "./images/test02.jpg",
"./images/test03.jpg", "./images/test04.jpg", "./images/test05.jpg",
"../images/test06.jpg", "./images/test07.jpg", "./images/test08.jpg", "./images/test09.jpg", "./images/test10.jpg"};
char
*resultsiamge[] =
{"result01.jpg","result02.jpg","result03.jpg","result04.jpg","result05.jpg","result06.jpg","result07.jpg",
"result08.jpg","result09.jpg","result10.jpg"};
int main( int argc, char** argv )
{

     int i = 0;
     char filename[80];
     storage = cvCreateMemStorage(0);
     
#ifdef DEBUG
     cvNamedWindow( "Template1", 1 );
     cvNamedWindow( "Template2", 1 );
     cvNamedWindow( "Comp Results1", 1 );
     cvNamedWindow( "Comp Results2", 1 );
     cvNamedWindow( "Skin Detection", 1 );
#endif
     openHandTmpl = cvLoadImage("openHandTmpl.jpg",1);
     closedHandTmpl = cvLoadImage("closedHandTmpl.jpg",1);
     
     // Convert form RGB to GRAY
     tmplSize.width = openHandTmpl->width;
     tmplSize.height = openHandTmpl->height;
     
     openGrayHandTmpl = cvCreateImage( tmplSize, IPL_DEPTH_8U, 1);
     cvCvtColor( openHandTmpl, openGrayHandTmpl, CV_BGR2GRAY );
     
     tmplSize.width = closedHandTmpl->width;
     tmplSize.height = closedHandTmpl->height;
     
     closedGrayHandTmpl = cvCreateImage( tmplSize, IPL_DEPTH_8U, 1);
     cvCvtColor( closedHandTmpl, closedGrayHandTmpl, CV_BGR2GRAY );
     cvNamedWindow("source",1);
     cvMoveWindow("source",0,0);

     cvNamedWindow( "Results", 1 );
     cvMoveWindow("Results",400,300);
     cvAddSearchPath("\\images\\");

     // Open the Output File
     output = fopen( "results.log", "w" );

     // Color for contour drawing
     color = CV_RGB( 255,255,255 );
     
     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 != 0)
     {
           for(;;)
           {
                 IplImage *frame;//, *frame_copy;
                 if( !cvGrabFrame( capture ))
                       break;
                 frame = cvRetrieveFrame( capture );
                 if( !frame )
                       break;
                 imgSize.width = frame->width;
                 imgSize.height = frame->height;

//       frame_copy = cvCloneImage( frame );
                 handDetect( frame );
                 if( cvWaitKey( 10 ) >= 0 )
                       break;
           }
           
           cvReleaseCapture( &capture );
     }
     else
     {
           
           for ( i = 0; i < numImg; i++ )
           {
                 fprintf(output, "Image %i %s ************************\n", i+1, names);
                 strcpy(filename, "result_");
//                  cvNamedWindow( names, 1 );
//                  cvNamedWindow("source",1);
//                  cvMoveWindow("source",0,0);
                 img = cvLoadImage(names,1);  
                 cvShowImage("source",img);
                 cvWaitKey(0);
                 if( img )
                 {
                       imgSize.width = img->width;
                       imgSize.height = img->height;
                       
                       handDetect( img );
//                        strcat(filename, names);
                       cvSaveImage(resultsiamge, img);
                       cvReleaseImage(&img);
                 }
           }
           cvWaitKey(0);
           
//            for ( i = 0; i < numImg; i++ )
//            {
//                  cvDestroyWindow( names);      
//            }
     }
     cvReleaseImage(&tmpImg);
     cvReleaseImage(&conv);
     cvReleaseImage(&H);
     cvReleaseImage(&S);
     cvReleaseImage(&V);
     cvReleaseImage(&tmpH1);
     cvReleaseImage(&tmpH2);
     cvReleaseImage(&tmpH3);
     cvReleaseImage(&tmpS1);
     cvReleaseImage(&tmpS2);
     cvReleaseImage(&tmpS3);
     cvReleaseImage(&openHandTmpl);
     cvReleaseImage(&closedHandTmpl);
     cvReleaseImage(&openGrayHandTmpl);
     cvReleaseImage(&closedGrayHandTmpl);
     cvReleaseImage(&openscaledTmpl);
     cvReleaseImage(&closedscaledTmpl);
     cvReleaseImage(&openMatchResult);
     cvReleaseImage(&closedMatchResult);
     cvReleaseMemStorage(&storage);
     fclose( output );
//      cvDestroyWindow( "Result");  
     
#ifdef DEBUG
     cvDestroyWindow( "Template1");
     cvDestroyWindow( "Template2" );
     cvDestroyWindow( "Comp Results1" );
     cvDestroyWindow( "Comp Results2" );
     cvDestroyWindow( "Skin Detection" );
#endif
     
     
     return 0;
     
}


void handDetect( IplImage *img)
{
// CvSize tmplSize;
     CvSize resultSize;
     IplConvKernel *erosionElement, *dilationElement;
     double contArea, imgArea, maxRatio = 0.0;
     CvRect bndRect = cvRect(0,0,0,0);
     CvPoint pt1, pt2;
     float scaleFactor;
     CvScalar averageValue;
     int startCount = 0;
     int openCount = 0;
     int closedCount = 0;
     
     float openCompRatio = 0.0;
     float closedCompRatio = 0.0;
     float bestRatio = 0.0;
     imgArea = imgSize.width * imgSize.height;

//       Open Template files
//      openHandTmpl = cvLoadImage("openHandTmpl.jpg",1);
//      closedHandTmpl = cvLoadImage("closedHandTmpl.jpg",1);
//      // Convert form RGB to GRAY
//      tmplSize.width = openHandTmpl->width;
//      tmplSize.height = openHandTmpl->height;
//      
//      openGrayHandTmpl = cvCreateImage( tmplSize, IPL_DEPTH_8U, 1);
//      cvCvtColor( openHandTmpl, openGrayHandTmpl, CV_BGR2GRAY );
//      
//      tmplSize.width = closedHandTmpl->width;
//      tmplSize.height = closedHandTmpl->height;
//      
//      closedGrayHandTmpl = cvCreateImage( tmplSize, IPL_DEPTH_8U, 1);
//      cvCvtColor( closedHandTmpl, closedGrayHandTmpl, CV_BGR2GRAY );

// Generate processing images
     tmpImg = cvCloneImage(img);
     conv = cvCreateImage( imgSize, IPL_DEPTH_8U, 3);
     tmpH1 = cvCreateImage( imgSize, IPL_DEPTH_8U, 1);
     tmpS1 = cvCreateImage( imgSize, IPL_DEPTH_8U, 1);
     tmpH2 = cvCreateImage( imgSize, IPL_DEPTH_8U, 1);
     tmpS2 = cvCreateImage( imgSize, IPL_DEPTH_8U, 1);
     tmpH3 = cvCreateImage( imgSize, IPL_DEPTH_8U, 1);
     tmpS3 = cvCreateImage( imgSize, IPL_DEPTH_8U, 1);
     H = cvCreateImage( imgSize, IPL_DEPTH_8U, 1);
     S = cvCreateImage( imgSize, IPL_DEPTH_8U, 1);
     V = cvCreateImage( imgSize, IPL_DEPTH_8U, 1);

     // Flip the image if in capture mode
     if ( capture )
     {
           cvFlip(conv, conv, 0);
     }


     // Image Smoothing
     //cvSmooth(img,tmpImg,CV_BLUR,3,3);
     cvSmooth(img,img,CV_GAUSSIAN,3,3);
     
     // Convert to HSV
     cvCvtColor( tmpImg, conv, CV_BGR2HSV );
     
     // Split to HSV planes
     cvCvtPixToPlane(conv,H,S,V,0);
     
     // Average Illumination
     averageValue = cvAvg( V , 0);
     fprintf(output, "Illumination Level = %f\n", averageValue.val[0]);
     
     // Detect skin tone Hues and Saturations
     // NOTE: Hue values need to be doubled for actual
     
     // Red to Orange Hue with High Saturation
     // Hue 0 to 28 degree and Sat 190 to 200
     cvInRangeS(H,cvScalar(0.0,0.0,0,0),cvScalar(14.0,0.0,0,0),tmpH1);
     cvInRangeS(S,cvScalar(75.0,0.0,0,0),cvScalar(200.,0.0,0,0),tmpS1);
     cvAnd(tmpH1,tmpS1,tmpH1,0);
     
     // Red Hue with Low Saturation
     // Hue 0 to 26 degree and Sat 20 to 90
     cvInRangeS(H,cvScalar(0.0,0.0,0,0),cvScalar(13.0,0.0,0,0),tmpH2);
     cvInRangeS(S,cvScalar(20.0,0.0,0,0),cvScalar(90.0,0.0,0,0),tmpS2);
     cvAnd(tmpH2,tmpS2,tmpH2,0);
     
     // Red Hue to Pink with Low Saturation
     // Hue 340 to 360 degree and Sat 15 to 90
     cvInRangeS(H,cvScalar(170.0,0.0,0,0),cvScalar(180.0,0.0,0,0),tmpH3);
     cvInRangeS(S,cvScalar(15.0,0.0,0,0),cvScalar(90.,0.0,0,0),tmpS3);
     cvAnd(tmpH3,tmpS3,tmpH3,0);
     
     // Combine the Hue and Sat detections
     cvOr(tmpH3,tmpH2,tmpH2,0);
     cvOr(tmpH1,tmpH2,tmpH1,0);
     

     // Dilation and Erosion
     
     // Structuring Element Generation
     dilationElement = cvCreateStructuringElementEx( 5,5,3,3, CV_SHAPE_RECT , 0 );
     erosionElement = cvCreateStructuringElementEx( 5,5,3,3, CV_SHAPE_RECT , 0 );
     
     // Dilation adds a layer on, and returns things to the correct size.
     cvDilate(tmpH1,tmpH2,dilationElement,1);
     
     // Erosion peels a layer of pixels off, and makes small regions disappear
     cvErode(tmpH1,tmpH3,erosionElement,1);
     
     // Find the contours of all remaining objects
     contour = 0;
     biggest = 0;
     cvFindContours( tmpH3, storage, &contour, sizeof(CvContour), CV_RETR_CCOMP, CV_CHAIN_APPROX_SIMPLE );
     cvZero(tmpH1);
     
     // contour now contains a CvSequence of all of the contours
     for( ; contour != 0; contour = contour->h_next )
     {
           contArea = fabs(cvContourArea( contour, CV_WHOLE_SEQ ));
           
           // Ignore very small contours
           if ( contArea/imgArea >= 0.015 )
           {
                 cvDrawContours( tmpH1, contour, color, color, 0, -1, 8 );
                 bndRect = cvBoundingRect( contour, 0 );
                 
                 // Scale the templates and result images
                 scaleFactor = ((float)bndRect.width / (float)openHandTmpl->width);
                 tmplSize.width = scaleFactor * openHandTmpl->width;
                 tmplSize.height = scaleFactor * openHandTmpl->height;
                 openscaledTmpl = cvCreateImage( tmplSize, IPL_DEPTH_8U, 1);
                 closedscaledTmpl = cvCreateImage( tmplSize, IPL_DEPTH_8U, 1);
                 openMatchResult = cvCreateImage( tmplSize, IPL_DEPTH_8U, 1);
                 closedMatchResult = cvCreateImage( tmplSize, IPL_DEPTH_8U, 1);
                 cvResize( openGrayHandTmpl,openscaledTmpl, CV_INTER_LINEAR );
                 cvResize( closedGrayHandTmpl,closedscaledTmpl, CV_INTER_LINEAR );
                 
                 // Set ROI for image
                 bndRect.width = tmplSize.width;
                 bndRect.height = tmplSize.height;
                 cvSetImageROI(tmpH1, bndRect);
                 
                 // Check that ROI matches the tmplate sizes
                 // If the ROI is too near the edge og the image then the
                 // ROI may be truncated when it is set and will not match the bndRect
                 if ( tmpH1->roi->width != bndRect.width || tmpH1->roi->height != bndRect.height )
                 {
                 }
                 else
                 {
                       // Compare the Templates to the image ROI
                       cvCmp( tmpH1,openscaledTmpl, openMatchResult, CV_CMP_EQ );
                       cvCmp( tmpH1,closedscaledTmpl, closedMatchResult, CV_CMP_EQ );
                 }
                 
                 // Analyze the results
                 startCount = cvCountNonZero( tmpH1 );
                 openCount = cvCountNonZero( openMatchResult );
                 closedCount = cvCountNonZero( closedMatchResult );
                 
                 openCompRatio = (float)openCount/(float)startCount;
                 closedCompRatio = (float)closedCount/(float)startCount;
                 bestRatio = 0.0;
                 
                 // Small regions may give eroneous results
                 if ( openCompRatio > 1.0 ) openCompRatio = 0.0;
                 if ( closedCompRatio > 1.0 ) closedCompRatio = 0.0;
                 
                 // Compare the Open and closed hand results
                 if ( openCompRatio > closedCompRatio )
                 {
                       if ( openCompRatio > maxRatio )
                       {
                             maxRatio = openCompRatio;
                             biggest = contour;
                             isHandOpen = 1;
                       }
                 }
                 else
                 {
                       if ( closedCompRatio > maxRatio )
                       {
                             maxRatio = closedCompRatio;
                             biggest = contour;
                             isHandOpen = 0;
                       }
                 }
                 // Reset ROI
                 cvResetImageROI( tmpH1 );
                 
                 // Output the results
                 fprintf(output, "Contour Non-zero Count = %i\n", startCount);
                 fprintf(output, "Open Hand Matched Non-zero Count   = %i Open Hand Ratio   = %f\n", openCount, openCompRatio);
 
               fprintf(output, "Clased Hand Matched Non-zero Count = %i
Clased Hand Ratio = %f\n\n", closedCount, closedCompRatio);
                 
#ifdef DEBUG
                 cvShowImage("Comp Results1",openMatchResult);
                 cvShowImage("Comp Results2",closedMatchResult);
                 cvShowImage("Template1",openscaledTmpl);
                 cvShowImage("Template2",closedscaledTmpl);
                 cvShowImage("Skin Detection",tmpH1);
                 cvWaitKey(0);
#endif


           }
           cvZero(tmpH1);
     }
     
     // Draw Detection Rectangle
     // Red for Open Hand
     // Green for Closed Hand
     if ( biggest && maxRatio > 0.60 )
     {
           bndRect = cvBoundingRect( biggest, 0 );
           cvResetImageROI( img );
           pt1.x = bndRect.x;
           pt1.y = bndRect.y;
           pt2.x = bndRect.x + bndRect.width;
           pt2.y = bndRect.y + bndRect.height;
           
           if ( isHandOpen )
           {
                 cvRectangle( img, pt1, pt2, CV_RGB(255,0,0), 5 );
           }
           else
           {
                 cvRectangle( img, pt1, pt2, CV_RGB(0,255,0), 5 );
           }
     }
     
     // Display the results
     cvShowImage("Results",img);
     cvWaitKey(0);
     // If we found any contours then free the memory they use.
     if (contour!=NULL)
     {
           cvClearSeq(contour);
     }
     cvClearMemStorage(storage);
     cvReleaseImage(&tmpImg);
     cvReleaseImage(&conv);
     cvReleaseImage(&tmpH1);
     cvReleaseImage(&tmpS1);
     cvReleaseImage(&tmpH2);
     cvReleaseImage(&tmpS2);
     cvReleaseImage(&tmpH3);
     cvReleaseImage(&tmpS3);
     cvReleaseImage(&H);
     cvReleaseImage(&S);
     cvReleaseImage(&V);
     cvReleaseImage(&openscaledTmpl);
     cvReleaseImage(&closedscaledTmpl);
     cvReleaseImage(&openMatchResult);
     cvReleaseImage(&closedMatchResult);
}

#ifdef _EiC
main(1,"HandDetect.c");
#endif
PARTNER CONTENT

文章评论2条评论)

登录后参与讨论

zhangshaobing517_935512703 2009-2-24 15:18

做什么的?

ash_riple_768180695 2009-2-24 13:21

http://www.dspdesignline.com/214502589
相关推荐阅读
zhangshaobing517_935512703 2011-03-21 01:28
KC24RT-300调试笔记
项目中需要使用LED驱动器,主要是为了让一串LED发出的光照一致,所以在试验中采用LED串联的方式比较好点,LED并联容易导致LED发光的 不均匀以及寿命减少。我在项目中采用金升阳公司的KC24RT-...
zhangshaobing517_935512703 2010-11-19 14:53
线程中CreateEvent和SetEvent及WaitForSingleObj
首先介绍CreateEvent是创建windows事件的意思,作用主要用在判断线程退出,程锁定方面.CreateEvent 函功能描述:创建或打开一个命名的或无名的事件对象.EVENT有两种状态:发信...
zhangshaobing517_935512703 2010-11-15 13:29
VS2008 BEGIN
Visual Studio 2008环境与VC6.0的环境存在着比较大的区别,下面就一些小小的区别在这里做一些探讨,欢迎指教!1、如果是调试控制台程序,很多时候点击“启动调试”后是一闪而过,此时可有两...
zhangshaobing517_935512703 2010-11-01 20:38
使用MFC的数组类
 MFC的数组类支持的数组类似于C++中的常规数组,可以存放任何数据类型。C++的常规数组在使用前必须将其定义成能够容纳所有可能需要的元素,而MFC数组类创建的对象可以根据需要动态地增大或减小,数组的...
zhangshaobing517_935512703 2010-09-07 13:14
循环
 循环设计的注意的事情:(1)双重循环的跳出问题,break只挑出所在的循环,如果使用双层FOR循环,单个BREAK就不可能跳出所有的双层(2)在迭代的时候,注意起始和终止的条件,尤其是终止问题(3)...
zhangshaobing517_935512703 2010-09-02 01:09
图像处理改进
1.特征点提取的算法  标志点的提取算法对结果的影响虽然没有经过试验或者计算的推算,每1个pix的偏差对结果的影响有多大,但是不可避免的,要想获得高精度的  测量结果,高精度的提取对结果的影响还是很大...
EE直播间
更多
我要评论
2
9
关闭 站长推荐上一条 /1 下一条