利用OpenCV的人脸检测给头像带上圣诞帽
我们来看下效果
原图:
效果:
原理其实很简单:
采用一张圣诞帽的png图像作为素材,
利用png图像背景是透明的,贴在背景图片上就是戴帽子的效果了。
人脸检测的目的主要是为了确定贴帽子的位置,类似ps中自由变换的功能,检测到人脸中间的位置,resize圣诞帽子和人脸大小匹配,确定位置,贴上去,ok!
代码:非常简洁,根据参考博客给出的代码,由OpenCV自带的人脸检测代码经过简单修改即可。
- // getheader.cpp : 定义控制台应用程序的入口点。
- //
- #include "stdafx.h"
- #include "opencv2/objdetect/objdetect.hpp"
- #include "opencv2/highgui/highgui.hpp"
- #include "opencv2/imgproc/imgproc.hpp"
- #include <iostream>
- #include <stdio.h>
- using namespace std;
- using namespace cv;
- #pragma comment(lib,"opencv_core2410d.lib")
- #pragma comment(lib,"opencv_highgui2410d.lib")
- #pragma comment(lib,"opencv_objdetect2410d.lib")
- #pragma comment(lib,"opencv_imgproc2410d.lib")
- /** Function Headers */
- void detectAndDisplay( Mat frame );
- /** Global variables */
- //-- Note, either copy these two files from opencv/data/haarscascades to your current folder, or change these locations
- String face_cascade_name = "D:\\Program Files\\opencv\\sources\\data\\haarcascades\\haarcascade_frontalface_alt.xml";
- String eyes_cascade_name = "D:\\Program Files\\opencv\\sources\\data\\haarcascades\\haarcascade_eye_tree_eyeglasses.xml";
- CascadeClassifier face_cascade;
- CascadeClassifier eyes_cascade;
- string window_name = "Capture - Face detection";
- RNG rng(12345);
- const int FRAME_WIDTH = 1280;
- const int FRAME_HEIGHT = 240;
- /**
- * @function main
- */
- int main( void )
- {
- CvCapture* capture;
- //VideoCapture capture;
- Mat frame;
- //-- 1. Load the cascades
- if( !face_cascade.load( face_cascade_name ) ){ printf("--(!)Error loading\n"); return -1; };
- if( !eyes_cascade.load( eyes_cascade_name ) ){ printf("--(!)Error loading\n"); return -1; };
- frame = imread("19.jpg");//背景图片
- //-- 3. Apply the classifier to the frame
- if( !frame.empty() )
- { detectAndDisplay( frame ); }
- waitKey(0);
- return 0;
- }
- void mapToMat(const cv::Mat &srcAlpha, cv::Mat &dest, int x, int y)
- {
- int nc = 3;
- int alpha = 0;
- for (int j = 0; j < srcAlpha.rows; j++)
- {
- for (int i = 0; i < srcAlpha.cols*3; i += 3)
- {
- alpha = srcAlpha.ptr<uchar>(j)[i / 3*4 + 3];
- //alpha = 255-alpha;
- if(alpha != 0) //4通道图像的alpha判断
- {
- for (int k = 0; k < 3; k++)
- {
- // if (src1.ptr<uchar>(j)[i / nc*nc + k] != 0)
- if( (j+y < dest.rows) && (j+y>=0) &&
- ((i+x*3) / 3*3 + k < dest.cols*3) && ((i+x*3) / 3*3 + k >= 0) &&
- (i/nc*4 + k < srcAlpha.cols*4) && (i/nc*4 + k >=0) )
- {
- dest.ptr<uchar>(j+y)[(i+x*nc) / nc*nc + k] = srcAlpha.ptr<uchar>(j)[(i) / nc*4 + k];
- }
- }
- }
- }
- }
- }
- /**
- * @function detectAndDisplay
- */
- void detectAndDisplay( Mat frame )
- {
- std::vector<Rect> faces;
- Mat frame_gray;
- Mat hatAlpha;
- hatAlpha = imread("2.png",-1);//圣诞帽的图片
- cvtColor( frame, frame_gray, COLOR_BGR2GRAY );
- equalizeHist( frame_gray, frame_gray );
- //-- Detect faces
- face_cascade.detectMultiScale( frame_gray, faces, 1.1, 2, 0|CV_HAAR_SCALE_IMAGE, Size(30, 30) );
- for( size_t i = 0; i < faces.size(); i++ )
- {
- Point center( faces[i].x + faces[i].width/2, faces[i].y + faces[i].height/2 );
- // ellipse( frame, center, Size( faces[i].width/2, faces[i].height/2), 0, 0, 360, Scalar( 255, 0, 255 ), 2, 8, 0 );
- // line(frame,Point(faces[i].x,faces[i].y),center,Scalar(255,0,0),5);
- Mat faceROI = frame_gray( faces[i] );
- std::vector<Rect> eyes;
- //-- In each face, detect eyes
- eyes_cascade.detectMultiScale( faceROI, eyes, 1.1, 2, 0 |CV_HAAR_SCALE_IMAGE, Size(30, 30) );
- for( size_t j = 0; j < eyes.size(); j++ )
- {
- Point eye_center( faces[i].x + eyes[j].x + eyes[j].width/2, faces[i].y + eyes[j].y + eyes[j].height/2 );
- int radius = cvRound( (eyes[j].width + eyes[j].height)*0.25 );
- // circle( frame, eye_center, radius, Scalar( 255, 0, 0 ), 3, 8, 0 );
- }
- // if(eyes.size())
- {
- resize(hatAlpha,hatAlpha,Size(faces[i].width, faces[i].height),0,0,INTER_LANCZOS4);
- // mapToMat(hatAlpha,frame,center.x+2.5*faces[i].width,center.y-1.3*faces[i].height);
- mapToMat(hatAlpha,frame,faces[i].x,faces[i].y-0.8*faces[i].height);
- }
- }
- //-- Show what you got
- imshow( window_name, frame );
- imwrite("merry christmas.jpg",frame);
- }
下面是摄像头实时戴帽子,改下主函数就好了:
- int main( void )
- {
- CvCapture* capture;
- //VideoCapture capture;
- Mat frame;
- //-- 1. Load the cascades
- if( !face_cascade.load( face_cascade_name ) ){ printf("--(!)Error loading\n"); return -1; };
- if( !eyes_cascade.load( eyes_cascade_name ) ){ printf("--(!)Error loading\n"); return -1; };
- // frame = imread("19.jpg");//背景图片
- VideoCapture cap(0); //打开默认的摄像头号
- if(!cap.isOpened()) //检测是否打开成功
- return -1;
- Mat edges;
- //namedWindow("edges",1);
- for(;;)
- {
- Mat frame;
- cap >> frame; // 从摄像头中获取新的一帧
- detectAndDisplay( frame );
- //imshow("edges", frame);
- if(waitKey(30) >= 0) break;
- }
- //摄像头会在VideoCapture的析构函数中释放
- waitKey(0);
- return 0;
- }
我的系统的是win10 64位的系统,之前摄像头出来都是黑的,发现需要用vs2010配置一下x64版本方可使用,查了半天还是自己之前写的博客靠谱:
就是按照win7 x64来配置,完美运行
http://blog.csdn.net/wangyaninglm/article/details/16325283
效果:
参考文献:
http://blog.csdn.net/lonelyrains/article/details/50388999
http://docs.opencv.org/doc/tutorials/objdetect/cascade_classifier/cascade_classifier.html
我调试好的工程:
利用OpenCV的人脸检测给头像带上圣诞帽的更多相关文章
- 利用html5、websocket和opencv实现人脸检测
最近学习人脸识别相关的东西,在MFC下使用OpenCV做了一个简单的应用.训练需要较多的数据,windows应用程序终究还是不方便,于是想着做成CS模式:检测识别都放在服务器端,视频获取和显示都放在网 ...
- OpenCV实现人脸检测
OpenCV实现人脸检测(转载) 原文链接:https://www.cnblogs.com/mengdd/archive/2012/08/01/2619043.html 本文介绍最基本的用OpenC ...
- 【转载】opencv实现人脸检测
全文转载自CSDN的博客(不知道怎么将CSDN的博客转到博客园,应该没这功能吧,所以直接复制全文了),转载地址如下 http://blog.csdn.net/lsq2902101015/article ...
- OpenCV学习系列(一) Mac下OpenCV + xcode人脸检测实现
# OpenCV学习系列(一) Mac下OpenCV + xcode人脸检测实现 [-= 博客目录 =-] 1-学习目标 1.1-本章介绍 1.2-实践内容 1.3-相关说明 2-学习过程 2.1-环 ...
- OpenCV&Qt学习之四——OpenCV 实现人脸检测与相关知识整理
开发配置 OpenCV的例程中已经带有了人脸检测的例程,位置在:OpenCV\samples\facedetect.cpp文件,OpenCV的安装与这个例子的测试可以参考我之前的博文Linux 下编译 ...
- 基于OpenCv的人脸检测、识别系统学习制作笔记之三
1.在windows下编写人脸检测.识别系统.目前已完成:可利用摄像头提取图像,并将人脸检测出来,未进行识别. 2.在linux下进行编译在windows环境下已经能运行的代码. 为此进行了linux ...
- 【AdaBoost算法】基于OpenCV实现人脸检测Demo
一.关于检测算法 分类器训练: 通过正样本与负样本训练可得到分类器,opencv有编译好的训练Demo,按要求训练即可生成,这里我们直接使用其已经训练好的分类器检测: 检测过程: 检测过程很简单,可以 ...
- 基于Opencv的人脸检测及识别
一.实验目的:我这里完成的是,将8张人脸图片(4组,每组两张)存入库中,选取1张图片,程序识别出与其匹配的另一张. 这里介绍分三个步骤完成该工作,①程序读取摄像头.拍照 ②程序从电脑文档中读取图片 ...
- opencv 摄像头人脸检测
PYTHON ubuntu16.04 默认安装的Python版本2.7.12,当用pip install opencv-python 安装了opencv for python 3.3.0.10后,运行 ...
随机推荐
- Spring学习笔记5——注解方式AOP
第一步:注解配置业务类 使用@Component("Pservice")注解ProductService 类 package com.spring.service; import ...
- Python中模块之collections系列
collection系列功能介绍 1. 常用的集中类 1. Counter(计数器) 计数器的常用方法如下: 创建一个字典计数器 格式:collections.Counter(obj) 例如:prin ...
- Automap sqlalchemy.ext.automap 自动映射数据库表结构
from sqlalchemy.ext.automap import automap_base from sqlalchemy.orm import Session from sqlalchemy i ...
- Linux学习之CentOS(八)----文件与目录的默认权限与隐藏权限(转)
文件与目录的默认权限与隐藏权限 一个文件有若干个属性, 包括读写运行(r, w, x)等基本权限,及是否为目录 (d) 与文件 (-) 或者是连结档 (l) 等等的属性! 要修改属性的方法在前面也约略 ...
- Java基础学习(1)——反射
反射就是把Java类中的各种成分映射成相应的Java类(主要用于框架开发) 反射的基石–>Class类 Java程序中的各个类属于同一事物,描述这类事务的Java类名就是Class. Class ...
- aways on 配置部署(二)——配置域
前一篇中我们基本了解了配置aways on的三个步骤,本篇就具体讲解如何配置域. DNS的配置 上篇可以看到三台服务器的ip地址,网关,DNS等配置,其中sqlDNS服务器的dns为自己的ip地址,s ...
- python中修改字符串的几种方法
在Python中,字符串是不可变类型,即无法直接修改字符串的某一位字符.因此改变一个字符串的元素需要新建一个新的字符串.常见的修改方法有以下4种. 方法1:将字符串转换成列表后修改值,然后用join组 ...
- 焦点轮播图(tab轮播)
主要有两部分:1.列表导航(小图片) 2.展示区(大图片) 页面布局: HTML部分: <div class="s_conC"> ...
- python复杂网络库networkx:绘图draw
http://blog.csdn.net/pipisorry/article/details/54291831 networkx使用matplotlib绘制函数 draw(G[, pos, ax, h ...
- SQL Server AlwaysON从入门到进阶(2)——存储
本文属于SQL Server AlwaysON从入门到进阶系列文章 前言: 本节讲解关于SQL Server 存储方面的内容,相对于其他小节而言这节比较短.本节会提供一些关于使用群集或者非群集系统过程 ...