Detectar una cara y guardar la cara detectada en OpenCV
-
26-10-2019 - |
Pregunta
Estoy tratando de detectar la cara en la imagen y tratando de guardar la cara detectada como una imagen en OpenCV.
Tener algunos problemas en la función de detectores a continuación.
#include "stdafx.h"
#include <stdio.h>
#include <cv.h>
#include <highgui.h>
#include <cxcore.h>
CvHaarClassifierCascade *cascade;
CvMemStorage *storage;
void detectFaces( IplImage *img );
int _tmain(int argc, _TCHAR* argv[])
{
//CvCapture *capture;
IplImage *img;//*out;
int key = 0;
char *filename = "C:/OpenCV2.1/data/haarcascades/haarcascade_frontalface_alt.xml";
cascade = ( CvHaarClassifierCascade* )cvLoad( filename, 0, 0, 0 );
storage = cvCreateMemStorage( 0 );
img = cvLoadImage("Yurico.png");
assert( cascade && storage && img );
cvNamedWindow( "video:", 1 );
//cvNamedWindow( "video1:", 1 );
//out = detectFaces( img );
detectFaces( img );
cvWaitKey( 0 );
//cvShowImage( "video", out );
cvDestroyWindow( "video:" );
//cvDestroyWindow( "video1:" );
cvReleaseImage( &img );
cvReleaseHaarClassifierCascade( &cascade );
cvReleaseMemStorage( &storage );
return 0;
}
void detectFaces( IplImage *img )
{
int i;
CvRect *r;
CvSeq *faces = cvHaarDetectObjects(
img,
cascade,
storage,
1.1,
3,
0 /*CV_HAAR_DO_CANNY_PRUNNING*/,
cvSize( 40, 40) );
for( i = 0 ; i < ( faces ? faces->total : 0 ) ; i++ ) {
CvRect *r = ( CvRect* )cvGetSeqElem( faces, i );
cvRectangle( img,
cvPoint( r->x, r->y ),
cvPoint( r->x + r->width, r->y + r->height ),
CV_RGB( 255, 0, 0 ), 1, 8, 0 );
}
//cvShowImage( "video:", img );
cvSetImageROI(img, CvRect *r);
IplImage *img2 = cvCreateImage(cvGetSize(img),
img->depth,
img->nChannels);
cvSaveImage("Lakshmen.jpg",img2);
}
Tener un error diciendo esto:
Error 1 error C2664: 'cvSetImageROI' : cannot convert parameter 2 from 'CvRect *' to 'CvRect' c:\users\hp\documents\visual studio 2010\projects\facedetect\facedetect\facedetect.cpp 67 1 facedetect
Quiere salvar la región de interés en otra imagen. Cualquier corrección o mejoramiento me diga ..
Solución
Debe pasar un cvrect y no un cvrect*, por lo que no necesita el puntero (*) antes de r. Y dado que ya es un cvrect, debes escribir:
cvSetImageROI(img, &r);
Otros consejos
cvSetImageROI()
toma una cvRect
Como el segundo argumento, y lo usa como parámetro de entrada para recortar la imagen en esa área.
En otras palabras, necesitas crear un cvRect
con información válida. Puede hacerlo antes de llamar a la función o en línea:
cvSetImageROI(img_corr, cvRect(x_pos, y_pos, width, height));
También noté que en tu código, creas CvRect* r;
en al menos 3 ubicaciones diferentes dentro de la misma función. ¡Mala práctica! Consejo: Cree variables en su código en el momento en que las use, no antes de eso.
Solo reemplace el CvRect *r
como r=(CvRect*)cvGetSeqElem( faces,i)
y después de que el bucle escriba las dos líneas
cvSetImageROI(img, cvRect(r->x,r->y,r->width,r->height));
cvSaveImage("C1.jpg",img);