我们来看下效果

原图:

效果:

原理其实很简单:

采用一张圣诞帽的png图像作为素材,

利用png图像背景是透明的,贴在背景图片上就是戴帽子的效果了。

人脸检测的目的主要是为了确定贴帽子的位置,类似ps中自由变换的功能,检测到人脸中间的位置,resize圣诞帽子和人脸大小匹配,确定位置,贴上去,ok!

代码:非常简洁,根据参考博客给出的代码,由OpenCV自带的人脸检测代码经过简单修改即可。

  1. // getheader.cpp : 定义控制台应用程序的入口点。
  2. //
  3.  
  4. #include "stdafx.h"
  5.  
  6. #include "opencv2/objdetect/objdetect.hpp"
  7. #include "opencv2/highgui/highgui.hpp"
  8. #include "opencv2/imgproc/imgproc.hpp"
  9.  
  10. #include <iostream>
  11. #include <stdio.h>
  12.  
  13. using namespace std;
  14. using namespace cv;
  15.  
  16. #pragma comment(lib,"opencv_core2410d.lib")
  17. #pragma comment(lib,"opencv_highgui2410d.lib")
  18. #pragma comment(lib,"opencv_objdetect2410d.lib")
  19. #pragma comment(lib,"opencv_imgproc2410d.lib")
  20.  
  21. /** Function Headers */
  22. void detectAndDisplay( Mat frame );
  23.  
  24. /** Global variables */
  25. //-- Note, either copy these two files from opencv/data/haarscascades to your current folder, or change these locations
  26. String face_cascade_name = "D:\\Program Files\\opencv\\sources\\data\\haarcascades\\haarcascade_frontalface_alt.xml";
  27. String eyes_cascade_name = "D:\\Program Files\\opencv\\sources\\data\\haarcascades\\haarcascade_eye_tree_eyeglasses.xml";
  28. CascadeClassifier face_cascade;
  29. CascadeClassifier eyes_cascade;
  30. string window_name = "Capture - Face detection";
  31. RNG rng(12345);
  32.  
  33. const int FRAME_WIDTH = 1280;
  34. const int FRAME_HEIGHT = 240;
  35. /**
  36. * @function main
  37. */
  38. int main( void )
  39. {
  40. CvCapture* capture;
  41. //VideoCapture capture;
  42. Mat frame;
  43.  
  44. //-- 1. Load the cascades
  45. if( !face_cascade.load( face_cascade_name ) ){ printf("--(!)Error loading\n"); return -1; };
  46. if( !eyes_cascade.load( eyes_cascade_name ) ){ printf("--(!)Error loading\n"); return -1; };
  47.  
  48. frame = imread("19.jpg");//背景图片
  49.  
  50. //-- 3. Apply the classifier to the frame
  51. if( !frame.empty() )
  52. { detectAndDisplay( frame ); }
  53.  
  54. waitKey(0);
  55.  
  56. return 0;
  57. }
  58.  
  59. void mapToMat(const cv::Mat &srcAlpha, cv::Mat &dest, int x, int y)
  60. {
  61. int nc = 3;
  62. int alpha = 0;
  63.  
  64. for (int j = 0; j < srcAlpha.rows; j++)
  65. {
  66. for (int i = 0; i < srcAlpha.cols*3; i += 3)
  67. {
  68. alpha = srcAlpha.ptr<uchar>(j)[i / 3*4 + 3];
  69. //alpha = 255-alpha;
  70. if(alpha != 0) //4通道图像的alpha判断
  71. {
  72. for (int k = 0; k < 3; k++)
  73. {
  74. // if (src1.ptr<uchar>(j)[i / nc*nc + k] != 0)
  75. if( (j+y < dest.rows) && (j+y>=0) &&
  76. ((i+x*3) / 3*3 + k < dest.cols*3) && ((i+x*3) / 3*3 + k >= 0) &&
  77. (i/nc*4 + k < srcAlpha.cols*4) && (i/nc*4 + k >=0) )
  78. {
  79. dest.ptr<uchar>(j+y)[(i+x*nc) / nc*nc + k] = srcAlpha.ptr<uchar>(j)[(i) / nc*4 + k];
  80. }
  81. }
  82. }
  83. }
  84. }
  85. }
  86.  
  87. /**
  88. * @function detectAndDisplay
  89. */
  90. void detectAndDisplay( Mat frame )
  91. {
  92. std::vector<Rect> faces;
  93. Mat frame_gray;
  94. Mat hatAlpha;
  95.  
  96. hatAlpha = imread("2.png",-1);//圣诞帽的图片
  97.  
  98. cvtColor( frame, frame_gray, COLOR_BGR2GRAY );
  99. equalizeHist( frame_gray, frame_gray );
  100. //-- Detect faces
  101. face_cascade.detectMultiScale( frame_gray, faces, 1.1, 2, 0|CV_HAAR_SCALE_IMAGE, Size(30, 30) );
  102.  
  103. for( size_t i = 0; i < faces.size(); i++ )
  104. {
  105.  
  106. Point center( faces[i].x + faces[i].width/2, faces[i].y + faces[i].height/2 );
  107. // ellipse( frame, center, Size( faces[i].width/2, faces[i].height/2), 0, 0, 360, Scalar( 255, 0, 255 ), 2, 8, 0 );
  108.  
  109. // line(frame,Point(faces[i].x,faces[i].y),center,Scalar(255,0,0),5);
  110.  
  111. Mat faceROI = frame_gray( faces[i] );
  112. std::vector<Rect> eyes;
  113.  
  114. //-- In each face, detect eyes
  115. eyes_cascade.detectMultiScale( faceROI, eyes, 1.1, 2, 0 |CV_HAAR_SCALE_IMAGE, Size(30, 30) );
  116.  
  117. for( size_t j = 0; j < eyes.size(); j++ )
  118. {
  119. Point eye_center( faces[i].x + eyes[j].x + eyes[j].width/2, faces[i].y + eyes[j].y + eyes[j].height/2 );
  120. int radius = cvRound( (eyes[j].width + eyes[j].height)*0.25 );
  121. // circle( frame, eye_center, radius, Scalar( 255, 0, 0 ), 3, 8, 0 );
  122. }
  123.  
  124. // if(eyes.size())
  125. {
  126. resize(hatAlpha,hatAlpha,Size(faces[i].width, faces[i].height),0,0,INTER_LANCZOS4);
  127. // mapToMat(hatAlpha,frame,center.x+2.5*faces[i].width,center.y-1.3*faces[i].height);
  128. mapToMat(hatAlpha,frame,faces[i].x,faces[i].y-0.8*faces[i].height);
  129. }
  130. }
  131. //-- Show what you got
  132. imshow( window_name, frame );
  133. imwrite("merry christmas.jpg",frame);
  134. }

下面是摄像头实时戴帽子,改下主函数就好了:

  1. int main( void )
  2. {
  3. CvCapture* capture;
  4. //VideoCapture capture;
  5. Mat frame;
  6.  
  7. //-- 1. Load the cascades
  8. if( !face_cascade.load( face_cascade_name ) ){ printf("--(!)Error loading\n"); return -1; };
  9. if( !eyes_cascade.load( eyes_cascade_name ) ){ printf("--(!)Error loading\n"); return -1; };
  10.  
  11. // frame = imread("19.jpg");//背景图片
  12.  
  13. VideoCapture cap(0); //打开默认的摄像头号
  14. if(!cap.isOpened()) //检测是否打开成功
  15. return -1;
  16.  
  17. Mat edges;
  18. //namedWindow("edges",1);
  19. for(;;)
  20. {
  21. Mat frame;
  22. cap >> frame; // 从摄像头中获取新的一帧
  23. detectAndDisplay( frame );
  24. //imshow("edges", frame);
  25. if(waitKey(30) >= 0) break;
  26. }
  27. //摄像头会在VideoCapture的析构函数中释放
  28. waitKey(0);
  29.  
  30. return 0;
  31. }

我的系统的是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的人脸检测给头像带上圣诞帽的更多相关文章

  1. 利用html5、websocket和opencv实现人脸检测

    最近学习人脸识别相关的东西,在MFC下使用OpenCV做了一个简单的应用.训练需要较多的数据,windows应用程序终究还是不方便,于是想着做成CS模式:检测识别都放在服务器端,视频获取和显示都放在网 ...

  2. OpenCV实现人脸检测

    OpenCV实现人脸检测(转载)  原文链接:https://www.cnblogs.com/mengdd/archive/2012/08/01/2619043.html 本文介绍最基本的用OpenC ...

  3. 【转载】opencv实现人脸检测

    全文转载自CSDN的博客(不知道怎么将CSDN的博客转到博客园,应该没这功能吧,所以直接复制全文了),转载地址如下 http://blog.csdn.net/lsq2902101015/article ...

  4. OpenCV学习系列(一) Mac下OpenCV + xcode人脸检测实现

    # OpenCV学习系列(一) Mac下OpenCV + xcode人脸检测实现 [-= 博客目录 =-] 1-学习目标 1.1-本章介绍 1.2-实践内容 1.3-相关说明 2-学习过程 2.1-环 ...

  5. OpenCV&Qt学习之四——OpenCV 实现人脸检测与相关知识整理

    开发配置 OpenCV的例程中已经带有了人脸检测的例程,位置在:OpenCV\samples\facedetect.cpp文件,OpenCV的安装与这个例子的测试可以参考我之前的博文Linux 下编译 ...

  6. 基于OpenCv的人脸检测、识别系统学习制作笔记之三

    1.在windows下编写人脸检测.识别系统.目前已完成:可利用摄像头提取图像,并将人脸检测出来,未进行识别. 2.在linux下进行编译在windows环境下已经能运行的代码. 为此进行了linux ...

  7. 【AdaBoost算法】基于OpenCV实现人脸检测Demo

    一.关于检测算法 分类器训练: 通过正样本与负样本训练可得到分类器,opencv有编译好的训练Demo,按要求训练即可生成,这里我们直接使用其已经训练好的分类器检测: 检测过程: 检测过程很简单,可以 ...

  8. 基于Opencv的人脸检测及识别

    一.实验目的:我这里完成的是,将8张人脸图片(4组,每组两张)存入库中,选取1张图片,程序识别出与其匹配的另一张. 这里介绍分三个步骤完成该工作,①程序读取摄像头.拍照 ②程序从电脑文档中读取图片   ...

  9. opencv 摄像头人脸检测

    PYTHON ubuntu16.04 默认安装的Python版本2.7.12,当用pip install opencv-python 安装了opencv for python 3.3.0.10后,运行 ...

随机推荐

  1. Spring学习笔记5——注解方式AOP

    第一步:注解配置业务类 使用@Component("Pservice")注解ProductService 类 package com.spring.service; import ...

  2. Python中模块之collections系列

    collection系列功能介绍 1. 常用的集中类 1. Counter(计数器) 计数器的常用方法如下: 创建一个字典计数器 格式:collections.Counter(obj) 例如:prin ...

  3. Automap sqlalchemy.ext.automap 自动映射数据库表结构

    from sqlalchemy.ext.automap import automap_base from sqlalchemy.orm import Session from sqlalchemy i ...

  4. Linux学习之CentOS(八)----文件与目录的默认权限与隐藏权限(转)

    文件与目录的默认权限与隐藏权限 一个文件有若干个属性, 包括读写运行(r, w, x)等基本权限,及是否为目录 (d) 与文件 (-) 或者是连结档 (l) 等等的属性! 要修改属性的方法在前面也约略 ...

  5. Java基础学习(1)——反射

    反射就是把Java类中的各种成分映射成相应的Java类(主要用于框架开发) 反射的基石–>Class类 Java程序中的各个类属于同一事物,描述这类事务的Java类名就是Class. Class ...

  6. aways on 配置部署(二)——配置域

    前一篇中我们基本了解了配置aways on的三个步骤,本篇就具体讲解如何配置域. DNS的配置 上篇可以看到三台服务器的ip地址,网关,DNS等配置,其中sqlDNS服务器的dns为自己的ip地址,s ...

  7. python中修改字符串的几种方法

    在Python中,字符串是不可变类型,即无法直接修改字符串的某一位字符.因此改变一个字符串的元素需要新建一个新的字符串.常见的修改方法有以下4种. 方法1:将字符串转换成列表后修改值,然后用join组 ...

  8. 焦点轮播图(tab轮播)

    主要有两部分:1.列表导航(小图片) 2.展示区(大图片) 页面布局: HTML部分:    <div class="s_conC">                  ...

  9. python复杂网络库networkx:绘图draw

    http://blog.csdn.net/pipisorry/article/details/54291831 networkx使用matplotlib绘制函数 draw(G[, pos, ax, h ...

  10. SQL Server AlwaysON从入门到进阶(2)——存储

    本文属于SQL Server AlwaysON从入门到进阶系列文章 前言: 本节讲解关于SQL Server 存储方面的内容,相对于其他小节而言这节比较短.本节会提供一些关于使用群集或者非群集系统过程 ...