• 基于虹软ArcFace2.0 java版 利用人脸识别猜年龄的demo


    首先感谢虹软,是你们提供这么好的SDK支撑了我们的想象力!

    这是一个用javav编写的可视化应用,用户通过自己的脸和计算机进行交互,计算机则通过萌萌女孩的语音和用户对话。
    核心程序就是利用ArcFace2.0识别性别、年龄,但是为了获得正面脸,会根据ArcFace2.0的人脸3D角度、用语音提醒用户,这是一个的互动环节。最后,程序会幽默的、萌萌的告诉用户他的性别、年龄。

    获取SDK [请戳这里](https://ai.arcsoft.com.cn/index.htm?utm_source=csdn&utm_medium=referral)

    完整的项目源码、可执行程序,放在百度网盘:链接: https://pan.baidu.com/s/1eHF66l111S3Rs0VaS7v_LA
    提取码: ffag

    其中主要的3个java文件,代码如下:

    ```
    =====================================
    HowOldAreU.java
    =====================================
    package app;

    import java.awt.EventQueue;
    import javax.swing.JFrame;
    import java.io.File;
    import java.io.FileNotFoundException;
    import java.io.RandomAccessFile;
    import java.nio.channels.FileChannel;
    import java.nio.channels.FileLock;
    import java.util.ArrayList;
    import java.util.List;
    import java.util.Map;
    import java.util.Timer;
    import java.util.TimerTask;
    import java.awt.BorderLayout;
    import com.alibaba.fastjson.JSONArray;
    import com.arcsoft.face.FaceEngine;
    import com.github.sarxos.webcam.Webcam;
    import com.github.sarxos.webcam.WebcamPanel;
    import tools.MyFunc;
    import javax.swing.JOptionPane;

    /*这是一个用javav编写的可视化应用,用户通过自己的脸和计算机进行交互,计算机则通过萌萌女孩的语音和用户对话。
    核心程序就是利用ArcFace2.0识别性别、年龄,但是为了获得正面脸,会根据ArcFace2.0的人脸3D角度、用语音提醒用户,这是一个的互动环节。
    最后,程序会幽默的、萌萌的告诉用户他的性别、年龄。

    作者:huanghua8080@126.com
    */

    public class HowOldAreU {

    //应用根目录
    public static String fs = File.separator;
    public final static String localPath = System.getProperty("user.dir")+fs;
    public final static String soundDir = localPath+"sound"+fs;
    //
    public static Webcam camera = null;
    private JFrame frame;
    //
    public static FaceEngine faceEngine = null;
    @SuppressWarnings("rawtypes")
    public static List FaceFeature = new ArrayList<Map<String, String>>();
    public static JSONArray aryFFTime = new JSONArray();
    public static JSONArray aryFFCnt = new JSONArray();
    public static String lastTime = "2019-01-09 13:30:00";
    public static int faceCnt = 0;

    /**
    * Launch the application.
    */
    public static void main(String[] args) {

    //判断程序是否已经运行
    String s = localPath+"lockApp.txt";
    //
    RandomAccessFile raf = null;
    try {
    raf = new RandomAccessFile(new File(s), "rws");
    } catch (FileNotFoundException e1) {
    JOptionPane.showMessageDialog(null, "独占文件时发生异常。"+e1, "错误",JOptionPane.ERROR_MESSAGE);
    System.exit(0);
    }
    FileChannel fcin = raf.getChannel();
    FileLock flin = null;
    try {
    flin = fcin.tryLock();
    } catch (Exception e) {
    JOptionPane.showMessageDialog(null, "锁文件时发生异常:"+e, "错误",JOptionPane.ERROR_MESSAGE);
    System.exit(0);
    }
    if (flin == null) {
    JOptionPane.showMessageDialog(null, "程序已在运行,不可重复。", "错误",JOptionPane.ERROR_MESSAGE);
    System.exit(0);
    }

    s = "D:\Dev\ec_workspace\cs1914age";
    if(!s.equals(System.getProperty("user.dir"))) {
    if(args.length == 0) {
    JOptionPane.showMessageDialog(null, "没有入参,程序将终止。", "错误",JOptionPane.ERROR_MESSAGE);
    System.exit(0);
    return;
    }
    if(!"age".equals(MyFunc.strTrim(args[0]).toLowerCase())) {
    JOptionPane.showMessageDialog(null, "入参错误,程序将终止。", "错误",JOptionPane.ERROR_MESSAGE);
    System.exit(0);
    return;
    }
    }

    //获取摄像头
    camera = Webcam.getDefault();
    if (camera == null) {
    JOptionPane.showMessageDialog(null, "摄像头获取失败。", "错误",JOptionPane.ERROR_MESSAGE);
    return;
    }

    //初始化人脸引擎
    s = HowOldAreUAs.initEngine();
    if(!"".equals(s)) {
    JOptionPane.showMessageDialog(null, s, "错误",JOptionPane.ERROR_MESSAGE);
    System.exit(0);
    return;
    }

    //启动窗体
    EventQueue.invokeLater(new Runnable() {
    public void run() {
    try {
    HowOldAreU window = new HowOldAreU();
    window.frame.setVisible(true);
    } catch (Exception e) {
    e.printStackTrace();
    }
    }
    });
    }

    /**
    * Create the application.
    */
    public HowOldAreU() {
    initialize();
    }

    /**
    * Initialize the contents of the frame.
    */
    private void initialize() {
    //
    frame = new JFrame();
    frame.setTitle("猜年龄");
    frame.setBounds(100, 100, 610, 370);
    frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
    frame.getContentPane().setLayout(new BorderLayout(0, 0));
    frame.setExtendedState(JFrame.MAXIMIZED_BOTH);
    frame.setUndecorated(true);//去边框

    //摄像头加载到面板
    WebcamPanel panel = new WebcamPanel(camera);
    frame.getContentPane().add(panel, BorderLayout.CENTER);

    //启动声音
    HowOldAreUAs.playSound(100);

    //线程(识别频率:毫秒)
    Timer timerMain = new Timer();
    timerMain.scheduleAtFixedRate(new TimerTask() {
    public void run() {
    if (camera != null) {
    HowOldAreUAs.photo();
    }
    }
    }, 0, 500);

    }

    }
    ```

    ```
    ====================================
    HowOldAreUAs
    ====================================
    package app;

    import java.awt.image.BufferedImage;
    import java.io.IOException;
    import java.math.BigDecimal;
    import java.util.ArrayList;
    import java.util.List;
    import java.util.Random;

    import com.alibaba.fastjson.JSONArray;
    import com.alibaba.fastjson.JSONObject;
    import com.arcsoft.face.AgeInfo;
    import com.arcsoft.face.Face3DAngle;
    import com.arcsoft.face.FaceFeature;
    import com.arcsoft.face.FaceInfo;
    import com.arcsoft.face.FaceSimilar;
    import com.arcsoft.face.FunctionConfiguration;
    import com.arcsoft.face.GenderInfo;
    import com.arcsoft.face.Rect;
    import com.arcsoft.face.enums.ImageFormat;
    import com.sun.jna.Platform;

    import app.FaceAbout.ImageInfo;
    import tools.MyFunc;
    import tools.SoundPlay;

    public class HowOldAreUAs {
    public static final int recoFreq = 60;//同一人不重复识别时间(秒)
    public static final int scoreThreshold = 70;//人脸相似度阀值
    //3D角度阀值
    public static final BigDecimal yes3d = new BigDecimal("5");

    //拍照
    @SuppressWarnings("unchecked")
    public static void photo() {
    int rtn=-1,sex=-1,age=-1;

    //当前时间
    String nowTime = MyFunc.getSvrTime("yyyy-MM-dd HH:mm:ss");

    //不重复识别时间(去除过期的)
    for(int n=HowOldAreU.aryFFTime.size()-1;n>=0;n--) {
    if(MyFunc.datetimeSub(HowOldAreU.aryFFTime.get(n).toString(), nowTime) >= recoFreq) {
    HowOldAreU.aryFFTime.remove(n);
    HowOldAreU.aryFFCnt.remove(n);
    HowOldAreU.FaceFeature.remove(n);
    }
    }

    //拍照
    BufferedImage cameraImg = HowOldAreU.camera.getImage();

    //找脸
    List<FaceInfo> faceInfoList = new ArrayList<FaceInfo>();
    ImageInfo imageInfo = new FaceAbout().bufferedImage2ImageInfo(cameraImg);
    HowOldAreU.faceEngine.detectFaces(imageInfo.getRgbData(), imageInfo.getWidth(), imageInfo.getHeight(), ImageFormat.CP_PAF_BGR24, faceInfoList);
    int cnt = faceInfoList.size();
    if (cnt == 0) {
    //5分钟后,如果没有人来,则呼唤
    if(MyFunc.datetimeSub(HowOldAreU.lastTime, nowTime) > 300) {
    HowOldAreU.lastTime = nowTime;
    playSound(200);
    }
    return;
    }
    HowOldAreU.lastTime = nowTime;

    //找最大脸(第一张脸即为最大脸)
    FaceInfo oneFace = faceInfoList.get(0);

    //提取脸纹
    FaceFeature CmFeature = new FaceFeature();
    rtn = HowOldAreU.faceEngine.extractFaceFeature(imageInfo.getRgbData(), imageInfo.getWidth(), imageInfo.getHeight(),
    ImageFormat.CP_PAF_BGR24, oneFace, CmFeature);
    if (rtn != 0) {
    playSound(250);
    return;
    }

    //是否刚刚识别过
    int rfe = 0;
    int dSimilScore = 0;
    FaceSimilar faceSimilar = new FaceSimilar();
    for(int n=0;n<HowOldAreU.aryFFTime.size();n++) {
    rtn = HowOldAreU.faceEngine.compareFaceFeature(CmFeature, (FaceFeature) HowOldAreU.FaceFeature.get(n), faceSimilar);
    if (rtn != 0) {
    return;
    }
    //得分
    dSimilScore = new BigDecimal(faceSimilar.getScore()).multiply(new BigDecimal("100")).setScale(0, BigDecimal.ROUND_HALF_UP).intValue();
    //大于阀值
    if(dSimilScore >= scoreThreshold){
    if(MyFunc.datetimeSub(HowOldAreU.aryFFTime.get(n).toString(), nowTime) < recoFreq) {
    rfe = 1;
    int hdt = Integer.parseInt( HowOldAreU.aryFFCnt.get(n).toString() );
    if(hdt >= 1 && hdt <= 3) {
    playSound(180+hdt);
    HowOldAreU.aryFFCnt.set(n, hdt+1 );
    //停顿一下
    try {
    Thread.sleep(3000);
    } catch(InterruptedException ex) {
    Thread.currentThread().interrupt();
    }
    }
    break;
    }
    }
    }
    //最近识别过
    if(rfe == 1) {return;}

    //识别过10个人后,做一次自我介绍
    if(HowOldAreU.faceCnt == 11) {
    HowOldAreU.faceCnt = 0;
    }
    if(HowOldAreU.faceCnt == 0) {
    playSound(150);
    HowOldAreU.faceCnt ++;
    }
    //停顿一下
    try {
    Thread.sleep(1000);
    } catch(InterruptedException ex) {
    Thread.currentThread().interrupt();
    }

    //原型
    faceInfoList.add(oneFace);
    rtn = HowOldAreU.faceEngine.process(imageInfo.getRgbData(), imageInfo.getWidth(), imageInfo.getHeight(),
    ImageFormat.CP_PAF_BGR24, faceInfoList,
    FunctionConfiguration.builder().supportAge(true).supportFace3dAngle(true).supportGender(true).build());
    if (rtn != 0) {
    playSound(250);
    return;
    }

    //3D信息提取
    List<Face3DAngle> face3DAngleList = new ArrayList<Face3DAngle>();
    rtn = HowOldAreU.faceEngine.getFace3DAngle(face3DAngleList);
    if (rtn != 0) {
    playSound(250);
    return;
    }
    if(face3DAngleList.size() == 0) {
    playSound(250);
    return;
    }

    //0: 正常,其他数值:检测结果不可信
    int status3d = face3DAngleList.get(0).getStatus();
    if(status3d != 0) {return;}
    BigDecimal pitch = new BigDecimal("0");
    BigDecimal roll = new BigDecimal("0");
    BigDecimal yaw = new BigDecimal("0");
    BigDecimal yes3db = new BigDecimal("0").subtract(yes3d);

    //俯仰角
    pitch = new BigDecimal(face3DAngleList.get(0).getPitch()).setScale(7, BigDecimal.ROUND_HALF_UP);
    if(pitch.compareTo(yes3d) == 1) {
    playSound(301);
    return;
    }
    if(pitch.compareTo(yes3db) == -1) {
    playSound(302);
    return;
    }
    //横滚角
    roll = new BigDecimal(face3DAngleList.get(0).getRoll()).setScale(7, BigDecimal.ROUND_HALF_UP);
    if(roll.compareTo(yes3d) == 1) {
    playSound(311);
    return;
    }
    if(roll.compareTo(yes3db) == -1) {
    playSound(312);
    return;
    }

    //偏航角
    yaw = new BigDecimal(face3DAngleList.get(0).getYaw()).setScale(7, BigDecimal.ROUND_HALF_UP);
    if(yaw.compareTo(yes3d) == 1) {
    playSound(321);
    return;
    }
    if(yaw.compareTo(yes3db) == -1) {
    playSound(322);
    return;
    }

    //年龄提取
    List<AgeInfo> ageInfoList = new ArrayList<AgeInfo>();
    rtn = HowOldAreU.faceEngine.getAge(ageInfoList);
    if (rtn != 0) {
    playSoundSexAge(-1,-1);
    return;
    }
    age = ageInfoList.get(0).getAge();
    if(age > 120) {age = 120;}

    //性别提取
    List<GenderInfo> genderInfoList = new ArrayList<GenderInfo>();
    rtn = HowOldAreU.faceEngine.getGender(genderInfoList);
    if (rtn != 0) {
    playSoundSexAge(-1,age);
    return;
    }
    sex = genderInfoList.get(0).getGender();

    //
    if(sex == -1 && age == -1) {
    playSound(360);
    return;
    }

    //播报
    playSoundSexAge(sex,age);

    //记录人脸,防止重复识别同一个人
    HowOldAreU.FaceFeature.add(CmFeature);
    HowOldAreU.aryFFTime.add(nowTime);
    HowOldAreU.aryFFCnt.add("1");

    //记录已识别数量
    HowOldAreU.faceCnt ++;
    //System.out.println(HowOldAreU.faceCnt+" "+now_time);
    }

    public static void playSoundSexAge(int sex,int age) {
    //不同年龄段,不同称谓
    String agename = "frend";
    if(sex >= 0) {
    if(age >= 0 && age <= 2) {
    agename = "00-"+sex;
    }else if(age >= 3 && age <= 18) {
    agename = "03-"+sex;
    }else if(age >= 19 && age <= 45) {
    agename = "19-"+sex;
    }else if(age >= 46 && age <= 75) {
    agename = "46-"+sex;
    }else if(age >= 76 && age <= 120) {
    agename = "76-"+sex;
    }
    }
    SoundPlay.playSoundFile(HowOldAreU.soundDir+"agename"+HowOldAreU.fs+"agename-"+agename+".mp3", null);

    //推测用语
    JSONArray ary = new JSONArray();
    ary.add("401");//你,大概
    ary.add("402");//我猜你
    ary.add("403");//我估计你
    ary.add("404");//我看你
    ary.add("405");//你看起来
    int cnt = ary.size();
    //随机选择一个
    int idx = 0;
    if(cnt > 1) {
    Random random = new Random();
    idx = random.nextInt(cnt)%(cnt+1);
    }
    SoundPlay.playSoundFile(HowOldAreU.soundDir+ary.get(idx)+".mp3", null);

    //年龄
    SoundPlay.playSoundFile(HowOldAreU.soundDir+"age"+HowOldAreU.fs+"age"+age+".mp3", null);

    //停顿一下
    try {
    Thread.sleep(1000);
    } catch(InterruptedException ex) {
    Thread.currentThread().interrupt();
    }

    //确认
    ary = new JSONArray();
    ary.add("481");//对不对啊?
    ary.add("482");//是不是呢?
    ary.add("483");//准吗?
    ary.add("484");//差不多吗?
    ary.add("485");//靠谱吧?
    cnt = ary.size();
    idx = 0;
    if(cnt > 1) {
    Random random = new Random();
    idx = random.nextInt(cnt)%(cnt+1);
    }
    SoundPlay.playSoundFile(HowOldAreU.soundDir+ary.get(idx)+".mp3", null);

    //停顿一下
    try {
    Thread.sleep(2000);
    } catch(InterruptedException ex) {
    Thread.currentThread().interrupt();
    }

    //笑一个
    ary = new JSONArray();
    ary.add("501");//哈哈!
    ary.add("502");//嘻嘻!
    cnt = ary.size();
    idx = 0;
    if(cnt > 1) {
    Random random = new Random();
    idx = random.nextInt(cnt)%(cnt+1);
    }
    SoundPlay.playSoundFile(HowOldAreU.soundDir+ary.get(idx)+".mp3", null);

    //如果错了
    ary = new JSONArray();
    ary.add("521");//如果我说错了,
    ary.add("522");//要是我没有说对,
    cnt = ary.size();
    idx = 0;
    if(cnt > 1) {
    Random random = new Random();
    idx = random.nextInt(cnt)%(cnt+1);
    }
    SoundPlay.playSoundFile(HowOldAreU.soundDir+ary.get(idx)+".mp3", null);

    //别生气
    ary = new JSONArray();
    ary.add("541");//你可别生气哦!
    ary.add("542");//你别往心里去啊!
    ary.add("543");//你千万别介意哈!
    cnt = ary.size();
    idx = 0;
    if(cnt > 1) {
    Random random = new Random();
    idx = random.nextInt(cnt)%(cnt+1);
    }
    SoundPlay.playSoundFile(HowOldAreU.soundDir+ary.get(idx)+".mp3", null);

    //停顿一下
    try {
    Thread.sleep(2000);
    } catch(InterruptedException ex) {
    Thread.currentThread().interrupt();
    }

    //下一个
    ary = new JSONArray();
    ary.add("561");//来,下一个!
    ary.add("562");//请下一位朋友!
    ary.add("563");//下一位,谁来?
    cnt = ary.size();
    idx = 0;
    if(cnt > 1) {
    Random random = new Random();
    idx = random.nextInt(cnt)%(cnt+1);
    }
    SoundPlay.playSoundFile(HowOldAreU.soundDir+ary.get(idx)+".mp3", null);
    }

    public static void playSound(int sound_kind) {
    // http://ai.baidu.com/tech/speech/tts
    JSONArray ary = new JSONArray();
    //文件集
    switch(sound_kind) {
    case 100:
    ary.add("101");//秋语已经启动,就要工作啦!
    break;

    case 150:
    //大家好,我是机器人。主人给我取名:秋语,他还帮我训练了一双火眼金睛,
    //看一眼就能识别出你们人类的性别和年龄。有人想过来试一试吗?
    ary.add("150");
    break;

    case 181:
    ary.add("181");//你来过的,一分钟之后再来,好吗?
    break;
    case 182:
    ary.add("182");//你来过的,一分钟之后再来,好吗?
    break;
    case 183:
    ary.add("183");//你怎么还来呀?跟你说了等一分钟的!你真是个急性子,不理你了。
    break;

    case 200: //没有发现人脸时
    ary.add("201");//怎么没有人来跟我玩儿?
    ary.add("202");//有人吗?快来和我玩啦!
    ary.add("203");//我知道你几岁了,过来试试吧!
    ary.add("204");//你们人呢?都到哪儿去了?
    ary.add("205");//我等了老半天,怎么连个人影也没看到!
    break;

    case 250: //看不请人脸或无法提取脸纹时
    ary.add("251");//嗨!靠近一点儿,我想看看你呢!
    ary.add("252");//喂!过来一点嘛,我都看不清你!
    ary.add("253");//hello,离我近一点儿,会有惊喜的!
    break;

    //3D角度过大
    case 301://俯仰角过大:请低一下头!
    case 302://俯仰角过大:把头抬一下!
    case 311://横滚角过大:头向左转一下!
    case 312://横滚角过大:向右转一下头!
    case 321://偏航角过大:脖子向左歪一下!
    case 322://偏航角过大:向右歪一下脖子!
    ary.add(sound_kind);
    break;

    case 360: //性别、年龄均未知
    ary.add("361");//你太神秘了,我实在猜不出你几岁!
    ary.add("362");//你到底几岁呢?我绞尽脑汁也想不出来!
    ary.add("363");//我无法识别你的年龄,我要请主人继续进化我。
    break;
    }
    int cnt = ary.size();
    if(cnt == 0) {return;}
    //随机选择一个
    int idx = 0;
    if(cnt > 1) {
    Random random = new Random();
    idx = random.nextInt(cnt)%(cnt+1);
    }
    //播放
    SoundPlay.playSoundFile(HowOldAreU.soundDir+ary.get(idx)+".mp3", null);
    }

    public static String initEngine() {
    JSONObject parm = MyFunc.GetAllProperties("config/parm.properties");
    String s = MyFunc.strTrim(parm.getString("err"));
    if(!"".equals(s)) {return "参数文件读取失败。"+s; }
    //APPID
    String APPID = MyFunc.strTrim(parm.getString("APP_ID"));
    if("".equals(APPID)){return "终端APPID缺失,程序将终止。";}
    //SDKKEY
    String WIN_SDKKEY = MyFunc.strTrim(parm.getString("WIN_SDKKEY"));
    String LIN_SDKKEY = MyFunc.strTrim(parm.getString("LIN_SDKKEY"));
    String SDKKEY = WIN_SDKKEY;
    if(!Platform.isWindows()) {SDKKEY = LIN_SDKKEY;}
    if("".equals(SDKKEY)){return "终端SDKKEY缺失,程序将终止。";}

    //加载动态库
    s = FaceAbout.loadDllSo();
    if(!"".equals(s)) {
    return "动态库加载失败,程序将终止。"+s;
    }
    //人脸引擎初始化
    try {
    HowOldAreU.faceEngine = FaceAbout.initFaceEngine(APPID, SDKKEY);
    } catch (IOException e) {
    return "人脸引擎初始化失败,程序将终止。"+e;
    }
    return "";
    }

    }

    ```

    ```
    ====================================
    FaceAbout.java
    ====================================
    package app;

    import java.awt.color.ColorSpace;
    import java.awt.image.BufferedImage;
    import java.awt.image.ColorConvertOp;
    import java.awt.image.DataBufferByte;
    import java.io.File;
    import java.io.IOException;
    import java.util.ArrayList;

    import com.arcsoft.face.EngineConfiguration;
    import com.arcsoft.face.FaceEngine;
    import com.arcsoft.face.FunctionConfiguration;
    import com.sun.jna.Platform;

    import tools.FileOpe;

    public class FaceAbout {

    //加载本地库
    public static String loadDllSo() {
    String fileType = "",s="",s2="";
    //路径
    String dllPath = HowOldAreU.localPath+"ArcSoft";
    //添加到系统路径
    addDirToPath(dllPath);

    //处理依赖库
    if(Platform.isWindows()) {
    String[] files = new String[2];
    files[0] = "msvcp120.dll";
    files[1] = "msvcr120.dll";
    for(int i=0;i<files.length;i++) {
    s = dllPath+HowOldAreU.fs+files;
    s2 = "C:\Windows\System32\"+files;
    if(!new File(s2).exists()) {
    try {
    new FileOpe().copyFile(new File(s), new File(s2));
    } catch (IOException e) {
    return "依赖库复制失败("+s+" -> "+s2+")。"+e;
    }
    }
    }
    }

    //文件类型
    if(Platform.isWindows()) {
    fileType = ".dll";
    }else {
    fileType = ".so";
    }
    ArrayList<String> ary = new ArrayList<String>();
    ary.add("libarcsoft_face");
    ary.add("libarcsoft_face_engine");
    ary.add("libarcsoft_face_engine_jni");
    //加载
    for(int i=0;i<ary.size();i++) {
    s = dllPath+HowOldAreU.fs+ary.get(i)+fileType;
    try {
    System.load(s);
    }catch (Exception e) {
    return "动态库("+s+")加载失败。"+e;
    }
    }
    return "";
    }

    //添加路径
    private static void addDirToPath(String s) {
    try {
    //获取系统path变量对象
    java.lang.reflect.Field field = ClassLoader.class.getDeclaredField("sys_paths");
    //设置此变量对象可访问
    field.setAccessible(true);
    //获取此变量对象的值
    String[] path = (String[])field.get(null);
    //是否已经存在
    for(int i=0;i<path.length;i++) {
    if(s.equals(path)) {
    field.setAccessible(false);
    return;
    }
    }
    //创建字符串数组,在原来的数组长度上增加一个,用于存放增加的目录
    String[] tem = new String[path.length+1];
    //将原来的path变量复制到tem中
    System.arraycopy(path,0,tem,0,path.length);
    //将增加的目录存入新的变量数组中
    tem[path.length] = s;
    //将增加目录后的数组赋给path变量对象
    field.set(null,tem);
    } catch (Exception e) {
    e.printStackTrace();
    }
    }

    //初始化
    public static FaceEngine initFaceEngine(String APPID, String SDKKEY) throws IOException {
    //初始化人脸引擎
    FaceEngine faceEngine = new FaceEngine();
    faceEngine.active(APPID, SDKKEY);
    //引擎配置
    EngineConfiguration engineConfiguration = EngineConfiguration.builder().functionConfiguration(
    FunctionConfiguration.builder()
    .supportAge(true)
    .supportFace3dAngle(true)
    .supportFaceDetect(true)
    .supportFaceRecognition(true)
    .supportGender(true)
    .build()).build();
    //初始化引擎
    faceEngine.init(engineConfiguration);
    return faceEngine;
    }

    //错误码
    public static String errCodeList(int codeNo) {
    String infoStr = "";
    switch(codeNo) {
    case 0: infoStr = "成功";break;
    case 1: infoStr = "错误原因不明";break;
    case 2: infoStr = "无效的参数";break;
    case 3: infoStr = "引擎不支持";break;
    case 4: infoStr = "内存不足";break;
    case 5: infoStr = "状态错误";break;
    case 6: infoStr = "用户取消相关操作";break;
    case 7: infoStr = "操作时间过期";break;
    case 8: infoStr = "用户暂停操作";break;
    case 9: infoStr = "缓冲上溢";break;
    case 10: infoStr = "缓冲下溢";break;
    case 11: infoStr = "存贮空间不足";break;
    case 12: infoStr = "组件不存在";break;
    case 13: infoStr = "全局数据不存在";break;
    case 28673: infoStr = "无效的AppId";break;
    case 28674: infoStr = "无效的SDKkey";break;
    case 28675: infoStr = "AppId和SDKKey不匹配";break;
    case 28676: infoStr = "SDKKey和使用的SDK不匹配";break;
    case 28677: infoStr = "系统版本不被当前SDK所支持";break;
    case 28678: infoStr = "SDK有效期过期,需要重新下载更新";break;
    case 73729: infoStr = "无效的输入内存";break;
    case 73730: infoStr = "无效的输入图像参数";break;
    case 73731: infoStr = "无效的脸部信息";break;
    case 73732: infoStr = "当前设备无GPU可用";break;
    case 73733: infoStr = "待比较的两个人脸特征的版本不一致";break;
    case 81921: infoStr = "人脸特征检测错误未知";break;
    case 81922: infoStr = "人脸特征检测内存错误";break;
    case 81923: infoStr = "人脸特征检测格式错误";break;
    case 81924: infoStr = "人脸特征检测参数错误";break;
    case 81925: infoStr = "人脸特征检测结果置信度低";break;
    case 86017: infoStr = "Engine不支持的检测属性";break;
    case 86018: infoStr = "需要检测的属性未初始化";break;
    case 86019: infoStr = "待获取的属性未在process中处理过";break;
    case 86020: infoStr = "ROCESS不支持的检测属性,例如FR,有自己独立的处理函数";break;
    case 86021: infoStr = "无效的输入图像";break;
    case 86022: infoStr = "无效的脸部信息";break;
    case 90113: infoStr = "SDK激活失败,请打开读写权限";break;
    case 90114: infoStr = "SDK已激活";break;
    case 90115: infoStr = "SDK未激活";break;
    case 90116: infoStr = "detectFaceScaleVal不支持";break;
    case 90117: infoStr = "SDK版本不匹配";break;
    case 90118: infoStr = "设备不匹配";break;
    case 90119: infoStr = "唯一标识不匹配";break;
    case 90120: infoStr = "参数为空";break;
    case 90121: infoStr = "活体检测功能已过期";break;
    case 90122: infoStr = "版本不支持";break;
    case 90123: infoStr = "签名错误";break;
    case 90124: infoStr = "数据库插入错误";break;
    case 90125: infoStr = "唯一标识符校验失败";break;
    case 90126: infoStr = "颜色空间不支持";break;
    case 90127: infoStr = "图片宽度或高度不支持";break;
    case 90128: infoStr = "android.permission.READ_PHONE_STATE权限被拒绝";break;
    case 90129: infoStr = "激活数据被破坏,请删除激活文件,重新进行激活";break;
    case 94209: infoStr = "无法解析主机地址";break;
    case 94210: infoStr = "无法连接服务器";break;
    case 94211: infoStr = "网络连接超时";break;
    case 94212: infoStr = "网络未知错误";break;
    case 98305: infoStr = "无法连接激活码服务器";break;
    case 98306: infoStr = "服务器系统错误";break;
    case 98307: infoStr = "请求参数错误";break;
    case 98308: infoStr = "激活码正确,且未被使用,但和传入的APPID及APPKEY不匹配";break;
    case 98309: infoStr = "传入的KEY值虽然正确,但此KEY已经被激活";break;
    case 98310: infoStr = "KEY格式不对,一般来说是KEY错误或者未传入KEY值";break;
    }
    return infoStr;
    }

    //
    public ImageInfo bufferedImage2ImageInfo(BufferedImage image) {
    ImageInfo imageInfo = new ImageInfo();
    int width = image.getWidth();
    int height = image.getHeight();
    // 使图片居中
    width = width & (~3);
    height = height & (~3);
    imageInfo.width = width;
    imageInfo.height = height;
    //根据原图片信息新建一个图片缓冲区
    BufferedImage resultImage = new BufferedImage(width, height, BufferedImage.TYPE_3BYTE_BGR);
    //得到原图的rgb像素矩阵
    int[] rgb = image.getRGB(0, 0, width, height, null, 0, width);
    //将像素矩阵 绘制到新的图片缓冲区中
    resultImage.setRGB(0, 0, width, height, rgb, 0, width);
    //进行数据格式化为可用数据
    BufferedImage dstImage = new BufferedImage(image.getWidth(), image.getHeight(), BufferedImage.TYPE_3BYTE_BGR);
    if (resultImage.getType() != BufferedImage.TYPE_3BYTE_BGR) {
    ColorSpace cs = ColorSpace.getInstance(ColorSpace.CS_LINEAR_RGB);
    ColorConvertOp colorConvertOp = new ColorConvertOp(cs, dstImage.createGraphics().getRenderingHints());
    colorConvertOp.filter(resultImage, dstImage);
    } else {
    dstImage = resultImage;
    }
    //获取rgb数据
    imageInfo.rgbData = ((DataBufferByte) (dstImage.getRaster().getDataBuffer())).getData();
    return imageInfo;
    }
    //
    public class ImageInfo {
    public byte[] rgbData;
    public int width;
    public int height;

    public byte[] getRgbData() {
    return rgbData;
    }

    public void setRgbData(byte[] rgbData) {
    this.rgbData = rgbData;
    }

    public int getWidth() {
    return width;
    }

    public void setWidth(int width) {
    this.width = width;
    }

    public int getHeight() {
    return height;
    }

    public void setHeight(int height) {
    this.height = height;
    }
    }

    }
    ```
    ==============
    程序在Windows7 64位 可以运行起来。里面调用摄像头的包,链接:https://github.com/sarxos/webcam-capture

    程序之中的不当之处,还请社区的技术专家和各位同仁批评指正,谢谢。

  • 相关阅读:
    【bzoj1116】[POI2008]CLO 并查集
    【bzoj3831】[Poi2014]Little Bird 单调队列优化dp
    【bzoj4726】[POI2017]Sabota? 树形dp
    【bzoj4822/bzoj1935】[Cqoi2017]老C的任务/[Shoi2007]Tree 园丁的烦恼 树状数组
    【bzoj4884】[Lydsy2017年5月月赛]太空猫 dp
    【bzoj2500】幸福的道路 树形dp+倍增RMQ+二分
    【bzoj2424】[HAOI2010]订货 费用流
    【bzoj2662】[BeiJing wc2012]冻结 分层图Spfa
    【bzoj1195】[HNOI2006]最短母串 AC自动机+状态压缩+BFS最短路
    【bzoj4818】[Sdoi2017]序列计数 矩阵乘法
  • 原文地址:https://www.cnblogs.com/KEAIILINN/p/10487523.html
Copyright © 2020-2023  润新知