CSDN2013年度博客之星评选活动開始,本人有幸入围参加评选,假设博客中的文章对你有所帮助,请为柳峰投上宝贵一票,很感谢!
投票地址:http://vote.blog.csdn.net/blogstaritem/blogstar2013/lyq8479
在笔者的公众账号小q机器人(微信号:xiaoqrobot)中有一个很好玩的功能"人脸检測",它能够检測出用户发送的图片中有多少张人脸,而且还能分析出每张脸所相应的人种、性别和年龄。差点儿每天都有一些用户在使用“人脸检測”,该功能的趣味性和娱乐性在于能够让用户知道自己的长相与真实年龄是否相符,是否男(女)性化。本文将为读者介绍人脸检測应用的完整实现过程。
人脸检測属于人脸识别的范畴,它是一个复杂的具有挑战性的模式匹配问题,国内外很多组织、科研机构都在专门研究该问题。国内的Face++团队专注于研发人脸检測、识别、分析和重建技术,而且向广大开发人员开放了人脸识别API,本文介绍的“人脸检測”应用正是基于Face++ API进行开发。
Face++简单介绍
Face++是北京旷视科技有限公司旗下的人脸识别云服务平台,Face++平台通过提供云端API、离线SDK、以及面向用户的自主研发产品等形式,将人脸识别技术广泛应用到互联网及移动应用场景中。Face++为广大开发人员提供了简单易用的API,开发人员能够轻松搭建属于自己的云端身份认证、用户兴趣挖掘、移动体感交互、社交娱乐分享等多种类型的应用。
Face++提供的技术服务包含人脸检測、人脸分析和人脸识别,主要说明例如以下:
1)人脸检測:能够从图片中高速、准确的定位面部的关键区域位置,包含眉毛、眼睛、鼻子、嘴巴等。
2)人脸分析:能够从图片或实时视频流中分析出人脸的性别(精确度达96%)、年龄、种族等多种属性。
3)人脸识别:能够高速判定两张照片是否为同一个人,或者高速判定视频中的人像是否为某一位特定的人。
Face++的中文网址为http://cn.faceplusplus.com/,要使用Face++ API,须要注冊成为Face++开发人员,也就是要注冊一个Face++账号。注冊完成后,先创建应用,创建应用时须要填写“应用名称”、“应用描写叙述”、“APIserver”、“应用类型”和“应用平台”,读者能够依据实际情况填写。应用创建完成后,能够看到应用的具体信息,例如以下图所看到的。
上图中,最重要的是API KEY和API SECRET,在调用Face++提供的API时,须要传入这两个参数。
人脸检測API介绍
在Face++站点的“API文档”中,能够看到Face++提供的全部API,我们要使用的人脸检測接口是detect分类下的“/detection/detect”,它能够检測出给定图片(Image)中的全部人脸(Face)的位置和相应的面部属性,眼下面部属性包含性别(gender)、年龄(age)、种族(race)、微笑程度(smiling)、眼镜(glass)和姿势(pose)。
读者能够在http://cn.faceplusplus.com/uc/doc/home?id=69中了解到人脸检測接口的具体信息,该接口的请求地址例如以下:
- http://apicn.faceplusplus.com/v2/detection/detect?url=URL&api_secret=API_SECRET&api_key=API_KEY
调用上述接口,必须要传入参数api_key、api_secret和待检測的图片。当中,待检測的图片能够是URL,也能够是POST方式提交的二进制数据。在微信公众账号后台,接收用户发送的图片,得到的是图片的訪问路径(PicUrl),因此,在本例中,直接使用待检測图片的URL是最方便的。调用人脸检測接口返回的是JSON格式数据例如以下:
- {
- "face": [
- {
- "attribute": {
- "age": {
- "range": 5,
- "value": 23
- },
- "gender": {
- "confidence": 99.9999,
- "value": "Female"
- },
- "glass": {
- "confidence": 99.945,
- "value": "None"
- },
- "pose": {
- "pitch_angle": {
- "value": 17
- },
- "roll_angle": {
- "value": 0.735735
- },
- "yaw_angle": {
- "value": -2
- }
- },
- "race": {
- "confidence": 99.6121,
- "value": "Asian"
- },
- "smiling": {
- "value": 4.86501
- }
- },
- "face_id": "17233b4b1b51ac91e391e5afe130eb78",
- "position": {
- "center": {
- "x": 49.4,
- "y": 37.6
- },
- "eye_left": {
- "x": 43.3692,
- "y": 30.8192
- },
- "eye_right": {
- "x": 56.5606,
- "y": 30.9886
- },
- "height": 26.8,
- "mouth_left": {
- "x": 46.1326,
- "y": 44.9468
- },
- "mouth_right": {
- "x": 54.2592,
- "y": 44.6282
- },
- "nose": {
- "x": 49.9404,
- "y": 38.8484
- },
- "width": 26.8
- },
- "tag": ""
- }
- ],
- "img_height": 500,
- "img_id": "22fd9efc64c87e00224c33dd8718eec7",
- "img_width": 500,
- "session_id": "38047ad0f0b34c7e8c6efb6ba39ed355",
- "url": "http://cn.faceplusplus.com/wp-content/themes/faceplusplus.zh/assets/img/demo/1.jpg?v=4"
- }
这里仅仅对本文将要实现的“人脸检測”功能中主要用到的参数进行说明,参数解释例如以下:
1)face是一个数组,当一张图片中包含多张人脸时,全部识别出的人脸信息都在face数组中。
2)age中的value表示预计年龄,range表示误差范围。比如,上述结果中value=23,range=5,表示人的真实年龄在18岁至28岁左右。
3)gender中的value表示性别,男性为Male,女性为Female;gender中的confidence表示检測结果的可信度。
4)race中的value表示人种,黄色人种为Asian,白色人种为White,黑色人种为Black;race中的confidence表示检測结果的可信度。
5)center表示人脸框中心点坐标,能够将x用于计算人脸的左右顺序,即x坐标的值越小,人脸的位置越靠近图片的左側。
人脸检測API的用法
为了方便开发人员调用人脸识别API,Face++团队提供了基于Objective-C、Java(Android)、Matlab、Ruby、C#等多种语言的开发工具包,读者能够在Face++站点的“工具下载”版块下载相关的SDK。在本例中,笔者并不打算使用官方提供的SDK进行开发,主要原因例如以下:1)人脸检測API的调用比較简单,自己写代码实现也并不复杂;2)假设使用SDK进行开发,笔者还要花费大量篇幅介绍SDK的使用,这些并非本文的重点;3)自己写代码实现比較灵活。当图片中有多张人脸时,人脸检測接口返回的数据是无序的,开发人员能够依照实际使用需求进行排序,比如,将图片中的人脸依照从左至右的顺序进行排序。
编程调用人脸检測API
首先,要对人脸检測接口返回的结构进行封装,建立与之相应的Java对象。因为人脸检測接口返回的参数较多,笔者仅仅是将本例中须要用到的参数抽取出来,封装成Face对象,相应的代码例如以下:
与普通Java类不同的是,Face类实现了Comparable接口,并实现了该接口的compareTo()方法,这正是Java中对象排序的关键所在。112-119行代码是通过比較每一个Face的脸部中心点的横坐标来决定对象的排序方式,这样能够实现检測出的多个Face按从左至右的先后顺序进行排序。
接下来,是人脸检測API的调用及相关处理逻辑,笔者将这些实现全部封装在FaceService类中,该类的完整实现例如以下:
- package org.liufeng.course.service;
-
- import java.io.BufferedReader;
- import java.io.InputStream;
- import java.io.InputStreamReader;
- import java.net.HttpURLConnection;
- import java.net.URL;
- import java.util.ArrayList;
- import java.util.Collections;
- import java.util.List;
- import org.liufeng.course.pojo.Face;
- import net.sf.json.JSONArray;
- import net.sf.json.JSONObject;
-
-
-
-
-
-
-
- public class FaceService {
-
-
-
-
-
-
- private static String httpRequest(String requestUrl) {
- StringBuffer buffer = new StringBuffer();
- try {
- URL url = new URL(requestUrl);
- HttpURLConnection httpUrlConn = (HttpURLConnection) url.openConnection();
- httpUrlConn.setDoInput(true);
- httpUrlConn.setRequestMethod("GET");
- httpUrlConn.connect();
-
- InputStream inputStream = httpUrlConn.getInputStream();
- InputStreamReader inputStreamReader = new InputStreamReader(inputStream, "utf-8");
- BufferedReader bufferedReader = new BufferedReader(inputStreamReader);
-
- String str = null;
- while ((str = bufferedReader.readLine()) != null) {
- buffer.append(str);
- }
- bufferedReader.close();
- inputStreamReader.close();
-
- inputStream.close();
- inputStream = null;
- httpUrlConn.disconnect();
-
- } catch (Exception e) {
- e.printStackTrace();
- }
- return buffer.toString();
- }
-
-
-
-
-
-
-
- private static List<Face> faceDetect(String picUrl) {
- List<Face> faceList = new ArrayList<Face>();
- try {
-
- String queryUrl = "http://apicn.faceplusplus.com/v2/detection/detect?url=URL&api_secret=API_SECRET&api_key=API_KEY";
-
- queryUrl = queryUrl.replace("URL", java.net.URLEncoder.encode(picUrl, "UTF-8"));
- queryUrl = queryUrl.replace("API_KEY", "替换成自己的API Key");
- queryUrl = queryUrl.replace("API_SECRET", "替换成自己的API Secret");
-
- String json = httpRequest(queryUrl);
-
- JSONArray jsonArray = JSONObject.fromObject(json).getJSONArray("face");
-
- for (int i = 0; i < jsonArray.size(); i++) {
-
- JSONObject faceObject = (JSONObject) jsonArray.get(i);
-
- JSONObject attrObject = faceObject.getJSONObject("attribute");
-
- JSONObject posObject = faceObject.getJSONObject("position");
- Face face = new Face();
- face.setFaceId(faceObject.getString("face_id"));
- face.setAgeValue(attrObject.getJSONObject("age").getInt("value"));
- face.setAgeRange(attrObject.getJSONObject("age").getInt("range"));
- face.setGenderValue(genderConvert(attrObject.getJSONObject("gender").getString("value")));
- face.setGenderConfidence(attrObject.getJSONObject("gender").getDouble("confidence"));
- face.setRaceValue(raceConvert(attrObject.getJSONObject("race").getString("value")));
- face.setRaceConfidence(attrObject.getJSONObject("race").getDouble("confidence"));
- face.setSmilingValue(attrObject.getJSONObject("smiling").getDouble("value"));
- face.setCenterX(posObject.getJSONObject("center").getDouble("x"));
- face.setCenterY(posObject.getJSONObject("center").getDouble("y"));
- faceList.add(face);
- }
-
- Collections.sort(faceList);
- } catch (Exception e) {
- faceList = null;
- e.printStackTrace();
- }
- return faceList;
- }
-
-
-
-
-
-
-
- private static String genderConvert(String gender) {
- String result = "男性";
- if ("Male".equals(gender))
- result = "男性";
- else if ("Female".equals(gender))
- result = "女性";
-
- return result;
- }
-
-
-
-
-
-
-
- private static String raceConvert(String race) {
- String result = "黄色";
- if ("Asian".equals(race))
- result = "黄色";
- else if ("White".equals(race))
- result = "白色";
- else if ("Black".equals(race))
- result = "黑色";
- return result;
- }
-
-
-
-
-
-
-
- private static String makeMessage(List<Face> faceList) {
- StringBuffer buffer = new StringBuffer();
-
- if (1 == faceList.size()) {
- buffer.append("共检測到 ").append(faceList.size()).append(" 张人脸").append("\n");
- for (Face face : faceList) {
- buffer.append(face.getRaceValue()).append("人种,");
- buffer.append(face.getGenderValue()).append(",");
- buffer.append(face.getAgeValue()).append("岁左右").append("\n");
- }
- }
-
- else if (faceList.size() > 1 && faceList.size() <= 10) {
- buffer.append("共检測到 ").append(faceList.size()).append(" 张人脸,按脸部中心位置从左至右依次为:").append("\n");
- for (Face face : faceList) {
- buffer.append(face.getRaceValue()).append("人种,");
- buffer.append(face.getGenderValue()).append(",");
- buffer.append(face.getAgeValue()).append("岁左右").append("\n");
- }
- }
-
- else if (faceList.size() > 10) {
- buffer.append("共检測到 ").append(faceList.size()).append(" 张人脸").append("\n");
-
- int asiaMale = 0;
- int asiaFemale = 0;
- int whiteMale = 0;
- int whiteFemale = 0;
- int blackMale = 0;
- int blackFemale = 0;
- for (Face face : faceList) {
- if ("黄色".equals(face.getRaceValue()))
- if ("男性".equals(face.getGenderValue()))
- asiaMale++;
- else
- asiaFemale++;
- else if ("白色".equals(face.getRaceValue()))
- if ("男性".equals(face.getGenderValue()))
- whiteMale++;
- else
- whiteFemale++;
- else if ("黑色".equals(face.getRaceValue()))
- if ("男性".equals(face.getGenderValue()))
- blackMale++;
- else
- blackFemale++;
- }
- if (0 != asiaMale || 0 != asiaFemale)
- buffer.append("黄色人种:").append(asiaMale).append("男").append(asiaFemale).append("女").append("\n");
- if (0 != whiteMale || 0 != whiteFemale)
- buffer.append("白色人种:").append(whiteMale).append("男").append(whiteFemale).append("女").append("\n");
- if (0 != blackMale || 0 != blackFemale)
- buffer.append("黑色人种:").append(blackMale).append("男").append(blackFemale).append("女").append("\n");
- }
-
- buffer = new StringBuffer(buffer.substring(0, buffer.lastIndexOf("\n")));
- return buffer.toString();
- }
-
-
-
-
-
-
-
- public static String detect(String picUrl) {
-
- String result = "未识别到人脸,请换一张清楚的照片再试!";
- List<Face> faceList = faceDetect(picUrl);
- if (null != faceList) {
- result = makeMessage(faceList);
- }
- return result;
- }
-
- public static void main(String[] args) {
- String picUrl = "http://pic11.nipic.com/20101111/6153002_002722872554_2.jpg";
- System.out.println(detect(picUrl));
- }
- }
上述代码尽管多,但条理很清楚,并不难理解,所以笔者仅仅挑重点的进行解说,主要说明例如以下:
1)70行:参数url表示图片的链接,因为链接中存在特殊字符,作为参数传递时必须进行URL编码。请读者记住:无论是什么应用,调用什么接口,凡是通过GET传递的参数中可能会包含特殊字符,都必须进行URL编码,除了中文以外,特殊字符还包含等号“=”、与“&”、空格“ ”等。
2)76-97行:使用JSON-lib解析人脸检測接口返回的JSON数据,并将解析结果存入List中。
3)99行:对集合中的对象进行排序,使用Collections.sort()方法排序的前提是集合中的Face对象实现了Comparable接口。
4)146-203行:组装返回给用户的消息内容。考虑到公众平台的文本消息内容长度有限制,当一张图片中识别出的人脸过多,则仅仅返回一些汇总信息给用户。
5)211-219行:detect()方法是public的,提供给其它类调用。笔者能够在本地的开发工具中执行上面的main()方法,測试detect()方法的输出。
公众账号后台的实现
在公众账号后台的CoreService类中,须要对用户发送的消息类型进行推断,假设是图片消息,则调用人脸检測方法进行分析,假设是其它消息,则返回人脸检測的使用指南。CoreService类的完整代码例如以下:
- package org.liufeng.course.service;
-
- import java.util.Date;
- import java.util.Map;
- import javax.servlet.http.HttpServletRequest;
- import org.liufeng.course.message.resp.TextMessage;
- import org.liufeng.course.util.MessageUtil;
-
-
-
-
-
-
-
- public class CoreService {
-
-
-
- public static String processRequest(HttpServletRequest request) {
-
- String respXml = null;
- try {
-
- Map<String, String> requestMap = MessageUtil.parseXml(request);
-
- String fromUserName = requestMap.get("FromUserName");
-
- String toUserName = requestMap.get("ToUserName");
-
- String msgType = requestMap.get("MsgType");
-
-
- TextMessage textMessage = new TextMessage();
- textMessage.setToUserName(fromUserName);
- textMessage.setFromUserName(toUserName);
- textMessage.setCreateTime(new Date().getTime());
- textMessage.setMsgType(MessageUtil.RESP_MESSAGE_TYPE_TEXT);
-
-
- if (MessageUtil.REQ_MESSAGE_TYPE_IMAGE.equals(msgType)) {
-
- String picUrl = requestMap.get("PicUrl");
-
- String detectResult = FaceService.detect(picUrl);
- textMessage.setContent(detectResult);
- }
-
- else
- textMessage.setContent(getUsage());
-
- respXml = MessageUtil.textMessageToXml(textMessage);
- } catch (Exception e) {
- e.printStackTrace();
- }
- return respXml;
- }
-
-
-
-
- public static String getUsage() {
- StringBuffer buffer = new StringBuffer();
- buffer.append("人脸检測使用指南").append("\n\n");
- buffer.append("发送一张清楚的照片,就能帮你分析出种族、年龄、性别等信息").append("\n");
- buffer.append("快来试试你是不是长得太着急");
- return buffer.toString();
- }
- }
到这里,人脸检測应用就全部开发完成了,整个项目的完整结构例如以下:
执行结果例如以下:
笔者用自己的相片測试了两次,測试结果各自是26岁、30岁,这与笔者的实际年龄相差不大,可见,Face++的人脸检測精确度还是比較高的。为了添加人脸检測应用的趣味性和娱乐性,笔者忽略了年龄预计值的正负区间。读者能够充分发挥自己的想像力和创造力,使用Face++ API实现许多其他有用、有趣的功能。应用开发不是简单的接口调用!
转帖请注明本文出自柳峰的博客(http://blog.csdn.net/lyq8479),请尊重他人的辛勤劳动成果,谢谢!