一、RGB color space

检测代码如下:

void SkinRGB(IplImage* src,IplImage* dst)
{
//RGB颜色空间
//均匀照明:R>95,G>40,B>20,R-B>15,R-G>15,R>B%R
//侧向照明:R>200,G>210,B>170,R-B<=15,R>B,G>B int height = src->height, width = src->width, channel = src->nChannels, step = src->widthStep;
int b = , g = , r = ; cvZero(dst);
unsigned char* p_src = (unsigned char*)src->imageData;
unsigned char* p_dst = (unsigned char*)dst->imageData; for(int j = ; j < height; j++){
for(int i = ; i < width; i++){
if((p_src[j*step+i*channel+r] > 95 && p_src[j*step+i*channel+g] > 40 && p_src[j*step+i*channel+b] > 20 &&
(p_src[j*step+i*channel+r] - p_src[j*step+i*channel+b]) > 15 && (p_src[j*step+i*channel+r] - p_src[j*step+i*channel+g]) > ) ||
(p_src[j*step+i*channel+r] > 200 && p_src[j*step+i*channel+g] > 210 && p_src[j*step+i*channel+b] > 170 &&
(p_src[j*step+i*channel+r] - p_src[j*step+i*channel+b]) <= 15 && p_src[j*step+i*channel+r] > p_src[j*step+i*channel+b] &&
p_src[j*step+i*channel+g] > p_src[j*step+i*channel+b]))
p_dst[j*width+i]=;
}
}
}

二、二次多项式模式检测(RG color space)

void cvSkinRG(IplImage* rgb,IplImage* gray)
{
assert(rgb->nChannels==&&gray->nChannels==); const int R=;
const int G=;
const int B=; double Aup=-1.8423;
double Bup=1.5294;
double Cup=0.0422;
double Adown=-0.7279;
double Bdown=0.6066;
double Cdown=0.1766;

for (int h=;h<rgb->height;h++) {
unsigned char* pGray=(unsigned char*)gray->imageData+h*gray->widthStep;
unsigned char* pRGB=(unsigned char* )rgb->imageData+h*rgb->widthStep;
for (int w=;w<rgb->width;w++){
int s=pRGB[R]+pRGB[G]+pRGB[B];
double r=(double)pRGB[R]/s;
double g=(double)pRGB[G]/s;
double Gup=Aup*r*r+Bup*r+Cup;
double Gdown=Adown*r*r+Bdown*r+Cdown;
double Wr=(r-0.33)*(r-0.33)+(g-0.33)*(g-0.33);
if (g<Gup && g>Gdown && Wr>0.004){
*pGray=;
}
else{
*pGray=;
}
pGray++;
pRGB+=;
}
}
}

三、Ycrcb之cr分量+otsu阈值化

原理:       a.将RGB图像转换到YCrCb颜色空间,提取Cr分量图像

b.对Cr做自适应二值化处理(Ostu法)

void cvSkinOtsu(IplImage* src, IplImage* dst)
{
//Cr自适应阈值法
// IplImage* img_ycrcb=cvCreateImage(cvGetSize(src),IPL_DEPTH_8U,);
IplImage* img_cr=cvCreateImage(cvGetSize(src),IPL_DEPTH_8U,); cvCvtColor(src,img_ycrcb,CV_BGR2YCrCb);
cvSplit(img_ycrcb,,img_cr,,);
cvThresholdOtsu(img_cr,img_cr);
cvCopy(img_cr,dst); cvReleaseImage(&img_ycrcb);
cvReleaseImage(&img_cr);
} void cvThresholdOtsu(IplImage* src, IplImage* dst)
{
int height=src->height,width=src->width,threshold=;
double histogram[]={};
double average=0.0,max_variance=0.0,w=0.0,u=0.0;
IplImage* temp=cvCreateImage(cvGetSize(src),src->depth,);
if(src->nChannels!=)cvCvtColor(src,temp,CV_BGR2GRAY);
else cvCopy(src,temp); unsigned char* p_temp=(unsigned char*)temp->imageData; //计算灰度直方图
//
for(int j=;j<height;j++) {
for(int i=;i<width;i++) {
histogram[p_temp[j*width+i]]++;
}
}
for(int i=;i<;i++)histogram[i]=histogram[i]/(double)(height*width); //计算平局值
for(int i=;i<;i++)average+=i*histogram[i]; for(int i=;i<;i++) {
w+=histogram[i];
u+=i*histogram[i]; double t=average*w-u;
double variance=t*t/(w*(-w));
if(variance>max_variance) {
max_variance=variance;
threshold=i;
}
}
cvThreshold(temp,dst,threshold,,CV_THRESH_BINARY); cvReleaseImage(&temp);
}

四、OpenCV自带肤色检测类——CvAdaptiveSkinDetector

通过颜色阈值分割肤色部分,皮肤检测算法是在HSV空间进行。流程如下:

//构造函数
CvAdaptiveSkinDetector(int samplingDivider = , int morphingMethod = MORPHING_METHOD_NONE); /*
参数1:样本采样的间隔,默认情况下为1,即表示不进行降采样
参数2:图形学操作方式,即对用皮肤检测后的图像进行图形学操作。其取值有3种可能——MORPHING_METHOD_ERODE(只进行一次腐蚀操作);MORPHING_METHOD_ERODE_ERODE(连续进行2次腐蚀操作);MORPHING_METHOD_ERODE_DILATE(先进行一次腐蚀操作,后进行一次膨胀操作)
*/
virtual void process(IplImage *inputBGRImage, IplImage *outputHueMask);

/*
参数1:需要进行皮肤检测的输入图像
参数2:输出皮肤的掩膜图像——值为1代表该像素为皮肤,值为0代表非皮肤。
*/

PS:这个函数只有opencv的c版本的,因为CvAdaptiveSkinDetector这个类放在opencv源码里的contrib目录里,即表示比较新的但不成熟的算法。

#include <opencv\cv.h>
#include <opencv\highgui.h>
#include <contrib\contrib.hpp>
#include <core\core.hpp>
#include <imgproc\imgproc.hpp>
#include<time.h> int main()
{
CvCapture* capture=cvCreateCameraCapture();
cvNamedWindow("Input Video",);
cvNamedWindow("Output Video",); IplImage* img_src=NULL;
IplImage* input_img=NULL;
IplImage* output_mask=NULL;
IplImage* output_img=NULL; clock_t start,finish;
double duration; CvAdaptiveSkinDetector skin_detector(,CvAdaptiveSkinDetector::MORPHING_METHOD_ERODE_DILATE); //定义肤色检测算子 while()
{
img_src=cvQueryFrame(capture);
if(!img_src) break;
cvShowImage("Input Video",img_src);
if(input_img==NULL){
input_img=cvCreateImage(cvGetSize(img_src),img_src->depth,img_src->nChannels);
}
cvCopy(img_src,input_img); output_img=cvCreateImage(cvGetSize(img_src),img_src->depth,img_src->nChannels);
cvZero(output_img); if(output_mask==NULL){
output_mask=cvCreateImage(cvGetSize(img_src),img_src->depth,);
} //肤色检测
//
start=clock();
skin_detector.process(input_img,output_mask);
finish=clock();
duration=(double)(finish-start)/CLOCKS_PER_SEC;
printf("elapsed time :%.0f 毫秒\n",duration*); cvCopy(img_src,output_img,output_mask);
cvShowImage("Output Video",output_img); char c=cvWaitKey();
if(c==)break;
} cvReleaseCapture(&capture);
cvDestroyWindow("Video");
}

五、HSV检测

void cvSkinHSV(IplImage* src,IplImage* dst)
{
IplImage* hsv=cvCreateImage(cvGetSize(src),,);
//IplImage* cr=cvCreateImage(cvGetSize(src),8,1);
//IplImage* cb=cvCreateImage(cvGetSize(src),8,1);
cvCvtColor(src,hsv,CV_BGR2HSV);
//cvSplit(ycrcb,0,cr,cb,0); static const int V=;
static const int S=;
static const int H=; //IplImage* dst=cvCreateImage(cvGetSize(_dst),8,3);
cvZero(dst); for (int h=;h<src->height;h++) {
unsigned char* phsv=(unsigned char*)hsv->imageData+h*hsv->widthStep;
unsigned char* psrc=(unsigned char*)src->imageData+h*src->widthStep;
unsigned char* pdst=(unsigned char*)dst->imageData+h*dst->widthStep;
for (int w=;w<src->width;w++) {
if (phsv[H]>=&&phsv[H]<=)
{
memcpy(pdst,psrc,);
}
phsv+=;
psrc+=;
pdst+=;
}
}
//cvCopyImage(dst,_dst);
//cvReleaseImage(&dst);
}

OpenCV——肤色检测的更多相关文章

  1. Python人体肤色检测

    代码地址如下:http://www.demodashi.com/demo/12967.html Python人体肤色检测 概述 本文中的人体肤色检测功能采用 OpenCV 库实现, OpenCV是一个 ...

  2. Atitit 图像处理--图像分类 模式识别 肤色检测识别原理 与attilax的实践总结

    Atitit 图像处理--图像分类 模式识别 肤色检测识别原理 与attilax的实践总结 1.1. 五中滤镜的分别效果..1 1.2. 基于肤色的图片分类1 1.3. 性能提升2 1.4. --co ...

  3. opencv直线检测在c#、Android和ios下的实现方法

    opencv直线检测在c#.Android和ios下的实现方法 本文为作者原创,未经允许,不得转载 :原文由作者发表在博客园:http://www.cnblogs.com/panxiaochun/p/ ...

  4. OpenCv皮肤检测-HSV分离

    HSV皮肤检测 // 进行肤色检测 void SkinDetect(IplImage* src, IplImage* dst) { // 创建图像头 IplImage* hsv = cvCreateI ...

  5. OPENCV条形码检测与识别

    条形码是当前超市和部分工厂使用比较普遍的物品,产品标识技术,使用摄像头检测一张图片的条形码包含有两个步骤,第一是定位条形码的位置,定位之后剪切出条形码,并且识别出条形码对应的字符串,然后就可以调用网络 ...

  6. SSE图像算法优化系列十:简单的一个肤色检测算法的SSE优化。

    在很多场合需要高效率的肤色检测代码,本人常用的一个C++版本的代码如下所示: void IM_GetRoughSkinRegion(unsigned char *Src, unsigned char ...

  7. OpenCV矩形检测

    OpenCV矩形检测 需求:提取图像中的矩形,图像存在污染现象,即矩形区域不是完全规则的矩形. 思路一:轮廓法 OpenCV里提取目标轮廓的函数是findContours,它的输入图像是一幅二值图像, ...

  8. keras系列︱人脸表情分类与识别:opencv人脸检测+Keras情绪分类(四)

    引自:http://blog.csdn.net/sinat_26917383/article/details/72885715 人脸识别热门,表情识别更加.但是表情识别很难,因为人脸的微表情很多,本节 ...

  9. OpenCV绘制检测结果

    OpenCV绘制检测结果 opencv  rtcp  timestamp  一.介绍 由于在验证阶段,使用FPGA时我们的算法检测速度很慢,没法直接在主流上进行绘图,否则的话,主流就要等待算法很久才能 ...

随机推荐

  1. gitosis随记

    0.创建git用户 useradd -m git passwd git 1.安装脚本工具(gitosis依赖python) apt-get install python-setuptools 2.gi ...

  2. RegExp子模式- "()"

    读书笔记 把JavaScript权威指南拿出来瞅瞅,正巧看到了第十章 正则表达式的模式匹配 最初接触js的时候,基本上都是在做验证.什么数字验证.命名验证.身份证格式验证.电话号码验证.都是用正则表达 ...

  3. vc远程调试启动进程(非attach)

    被调试端设置同attach进程方式的远程调试 代码端,需要在[Project] [Properties] [Configuration Properties] [Debugging].将Debugge ...

  4. redis之lua脚本

    背景介绍 redis数据库提供了一些管理功能比如 流水线:打包发送多条命令,并在一个回复里面接收所有被执行命令的结果.事务:一次执行多条命令,被执行的命令要么就全部都被执行,要么就一个也不执行.并且事 ...

  5. 火狐的bug

    初次启动火狐的界面并且默认是最大化的情况下,第一个业签时会发现火狐的浏览器无法达到下边框,请看图 途中可以看到,body区域没有填充满浏览器可用区域.但是当浏览器已经启动页签,现在是第二个页签时,则不 ...

  6. 网页UI视觉设计规范

  7. httpcomponents-client-4.3.6 HttpPost的简单使用

    /** * httpcomponents-client-4.3.6 * @author y */ public class HttpUtil { public static String httpPo ...

  8. format %x invalid or incompatible with argument问题解决方法

    现在还有好多朋友在用Protel 99se来画图,可是在现在的双核或四核电脑上运行Protel出现错误并且弹出对话框:“format '%x' invalid or incompatible with ...

  9. 关于google的C++ coding style

    大家都知道google的开源项目有很多,不过我观察过一些开源项目,觉得代码质量就是这家最好了.这些“教条”式规定的背后是是来自于常年工程经验积累上的理性思考. 为什么好?主要有以下几点: 1.规范,就 ...

  10. 关于php-fpm通讯时没有REQUEST_METHOD的问题

    nginx是通过fastcgi协议来和php通讯的!而php-fpm就扮演了这样的角色 fastcgi协议 中文版http://blog.chinaunix.net/uid-380521-id-241 ...