之前写过用openCV识别人脸和训练模型,这次说说用模型文件来对比人脸

首先要调起本地摄像头,然后识别一下人脸,这个人脸框出来,再调用模型文件进行人脸对比,识别成功显示用户名在人脸框框上面

 

这次我把这个东东结合了另一个项目,写成了一个客户端,然后可以输入实验室ID和用户ID去进行一系列操作,客户端接口请求到另一个项目的服务端,通过服务端去访问Mysql数据库读取一些用户信息和实验室信息,并且校验是否有开门权限,以此来达到打开实验室门的作用

 

另外,一开始写的过程中,遇到一个脑壳疼的问题

就是我一请求服务端接口,java客户端的画面就会卡住不动,后来查老半天,也没查出是啥问题,有人说是UI线程问题

然后联想想到以前写app的时候,安卓要通过子线程去访问网络再通知主线程更新UI界面

于是我在java客户端起了一个子线程去访问服务端,然后问题就解决了哈哈

 

以下是相关代码,用到相关工具类可以参考我之前写的博客上面有贴,服务端接口是我自己另一个项目里面写的,这里没有

其他问题可以评论或私信留言

 

Recognition.java

package face.recognize;

import java.awt.Dimension;
import java.awt.FlowLayout;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.io.File;
import java.io.IOException;
import java.text.DecimalFormat;
import java.util.HashMap;
import java.util.Map;

import javax.swing.*;

import face.detect.FaceDetection;
import face.util.MyConstant;
import face.util.OkHttpHelper;
import face.util.VideoPanel;
import org.opencv.core.Core;
import org.opencv.core.Mat;
import org.opencv.core.MatOfRect;
import org.opencv.core.Point;
import org.opencv.core.Rect;
import org.opencv.core.Scalar;
import org.opencv.core.Size;
import org.opencv.face.FaceRecognizer;
import org.opencv.face.LBPHFaceRecognizer;
import org.opencv.imgproc.Imgproc;
import org.opencv.objdetect.CascadeClassifier;
import org.opencv.objdetect.Objdetect;
import org.opencv.videoio.VideoCapture;

public class Recognition {

    private static String userId = "";

    private static String labId = "";

    public final static String BASE_FILE_PATH = "/face/model/";

    public final static String BASE_FILE_NAME = "/face_model.yml";

    public static String getUserId() {
        return userId;
    }

    public static void setUserId(String userId) {
        Recognition.userId = userId;
    }

    public static String getLabId() {
        return labId;
    }

    public static void setLabId(String labId) {
        Recognition.labId = labId;
    }
	
	static int tag = 0;
	
	public static VideoPanel vp;
	
	

public static VideoPanel getVp() {
		return vp;
	}

	public static void setVp(VideoPanel vp) {
		Recognition.vp = vp;
	}

public static void main(String[] args) throws Exception {
	
  	if (userId.equals("") || labId.equals("")){
        JOptionPane.showMessageDialog(null,"账号或实验室编号不可为空!","提示", JOptionPane.ERROR_MESSAGE);
        throw new Exception("null error");
    }
//通过模型存放的文件名来获取用户对应的模型文件

      String readPath = BASE_FILE_PATH + userId + BASE_FILE_NAME;
      System.out.println("file path---------->" + readPath);
      File testFile = new File(readPath);
      if(!testFile .exists()) {
    	  JOptionPane.showMessageDialog(null,"模型文件不存在!请确认操作是否正确!","提示", JOptionPane.ERROR_MESSAGE);
          System.out.println("file path---------->模型文件不存在!请确认操作是否正确!");
          throw new Exception("error");
      }
    System.loadLibrary(Core.NATIVE_LIBRARY_NAME);

    CascadeClassifier faceCascade = new CascadeClassifier();
    faceCascade.load(MyConstant.BASE_LOCATION+"/face/install/etc/haarcascades/haarcascade_frontalface_alt.xml");
    FaceRecognizer faceRecognizer = LBPHFaceRecognizer.create();
    faceRecognizer.read(readPath);
    VideoCapture capture = new VideoCapture();
    try {
      capture.open(0);
      VideoPanel videoPanel = VideoPanel.show("Recognition", 640, 480, 0);
  	//对比识别过程
      if (capture.isOpened()) {
        Mat image = new Mat();
        while (true) {
          capture.read(image);
          if (!image.empty()) {
            detectAndDisplay(image, faceCascade, faceRecognizer);
            videoPanel.setImageWithMat(image);
//            setVp(videoPanel);
            if (tag == 1) {
            	tag = 3;
			JOptionPane.showMessageDialog(null, "开门成功!", "提示",JOptionPane.INFORMATION_MESSAGE);
            } else if(tag == 2) {
            	tag = 3;
			JOptionPane.showMessageDialog(null, "开门失败!请确认用户信息和预约信息!", "提示",JOptionPane.ERROR_MESSAGE);
            }
          } else {
            break;
          }
        }
      }
      
    } finally {
      capture.release();
    }
  }

  public static void detectAndDisplay(Mat frame, CascadeClassifier faceCascade,
      FaceRecognizer faceRecognizer) {
    MatOfRect faces = new MatOfRect();
    Mat grayFrame = new Mat();

    Imgproc.cvtColor(frame, grayFrame, Imgproc.COLOR_BGR2GRAY);
    Imgproc.equalizeHist(grayFrame, grayFrame);

    // detect faces
    faceCascade.detectMultiScale(grayFrame, faces);
    Imgproc.equalizeHist(grayFrame, grayFrame);

    Rect[] facesArray = faces.toArray();
    for (int i = 0; i < facesArray.length; i++) {
      int[] label = new int[1];
      double[] confidence = new double[1];
      faceRecognizer.predict(grayFrame.submat(facesArray[i]), label, confidence);
      String name = faceRecognizer.getLabelInfo(label[0]);
      Scalar color;
      if (confidence[0] < 50) {
        if (label[0] == 0) {
          color = new Scalar(255, 0, 0);
        } else if (label[0] == 1) {
          color = new Scalar(0, 255, 0);
        } else {
          color = new Scalar(0, 255, 255);
        }
        name = name + " " + new DecimalFormat("#.0").format(confidence[0]);
      } else {
        name = "face";
        color = new Scalar(0, 0, 255);
      }

      Imgproc.rectangle(frame, facesArray[i].tl(), facesArray[i].br(), color, 2);
      setLabel(frame, name, facesArray[i].tl(), color);
    }

  }
  
  
  static void setLabel(Mat im, String label, Point or, Scalar color) {
    int fontface = Core.FONT_HERSHEY_SIMPLEX;
    double scale = 0.8;
    int thickness = 2;
    int[] baseline = new int[1];

    Size text = Imgproc.getTextSize(label, fontface, scale, thickness, baseline);
    Imgproc.rectangle(im, new Point(or.x, or.y),
        new Point(or.x + text.width, or.y - text.height - baseline[0] - baseline[0]), color,
        Core.FILLED);
//    System.out.println("识别信息-------------->"+label);
    
    if(!label.equals("face") && tag==0) {
    	String[] sp = label.split(" ");
    	Map<String, Object> map = new HashMap<String, Object>();
//    	测试数据
//    	map.put("userId", "1508141221");
//    	map.put("labId", 1);
    	map.put("userId", userId);
    	map.put("labId", labId);
    	map.put("isTag", 1);
    	try {
			String res = OkHttpHelper.sendGet(MyConstant.BASE_URL, map);
			if(res.equals("success")) {
				tag = 1;
			} else if (res.equals("error")) {
				tag = 2;
			}
		} catch (IOException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
    }
    Imgproc.putText(im, label, new Point(or.x, or.y - baseline[0]), fontface, scale,
        new Scalar(255, 255, 255), thickness);
    
    
  }
  

}

OkHttpHelper.java

import java.io.IOException;
import java.util.Map;

import okhttp3.OkHttpClient;
import okhttp3.Request;
import okhttp3.Response;

/**
 * okhttp请求工具类,用于向服务端发送请求
 * @author swallow
 *
 */
public class OkHttpHelper {

	private static final int COM_SUCCESS = 1;
	
	private static final int COM_ERROR = 0;
	
    public static String sendGet(String url,Map<String, Object> map) throws IOException {
        OkHttpClient mClient = new OkHttpClient();
        String result = "";
        StringBuilder reqUrl = new StringBuilder(url);
        if (map != null && map.size() > 0) {
            reqUrl.append("?");
            for (Map.Entry<String, Object> param : map.entrySet()) {
                reqUrl.append(param.getKey() + "=" + param.getValue() + "&");

            }
            url = reqUrl.subSequence(0, reqUrl.length() - 1).toString();
            System.out.println("start sendGet----->"+ url);
        }

        Request request = new Request.Builder()
                .url(url)
                .get()
                .build();
        try (Response response = mClient.newCall(request).execute()) {
            result = response.body().string();
            System.out.println("result sendGet----->"+ result);
        }
        return result;
    }
	
}

客户端启动,这里我用JFrame做了个客户端,emmm客户端没有怎么学过,这个写的界面比较简单

package face.util;
import face.detect.FaceDetection;
import face.recognize.CollectData;
import face.recognize.Recognition;
import face.recognize.Train;

import javax.swing.*;
import java.awt.*;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.io.IOException;
import java.util.HashMap;
import java.util.Map;

public class Main {

    public static void main(String[] args)  {
//    	VideoPanel videoPanel = VideoPanel.show("Recognition", 640, 480);
    	mainShow();
    }
    
    private static void mainShow() {
    	Map<String,Object> cacheMap = new HashMap<String,Object>();
        System.out.println("-----------------start Face Main-----------------");
        JFrame fram = new JFrame();                           //新建生成一个窗口
        fram.setSize(1000,450);                 //设置窗口大小

        fram.setLocationRelativeTo(null);                   //相对于什么居中,不填默认桌面居中
        fram.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);//设置关闭按钮关闭窗口并退出,括号内的值是Jfram的属性值,可以查询需要什么

        //JPanel panel=new JPanel();                        //新建面板
//        JPanel panel=(JPanel)fram.getContentPane();         //获取默认面板 Container是JPanel父类  所以需要强制转换
        JPanel panel=new JPanel(); 
        panel.setLayout(new FlowLayout(FlowLayout.CENTER ));//设置该面板的布局方式
        panel.setSize(300, 450);
        
        JButton buttonFaceDetection = new JButton("开启人脸检测");         //新建按钮
        JButton buttonFaceCollectData = new JButton("开启人脸收集");
        JButton buttonFaceTrain = new JButton("训练模型");
        JButton buttonOpenLab = new JButton("刷脸进入实验室");

        JTextField tfUserId = new JTextField();                  //新建输入框
        JTextField tfLabId = new JTextField();

        String note = "<html>说明:第一次使用此程序必须使用人脸收集与训练模型,否则无法正常开门!" +
                "训练一次过后下次可不再训练,账号与实验室编号为必填项!</html>";
        JLabel jlNote = new JLabel();
        JLabel jlUserId = new JLabel("请输入账号");
        JLabel jlLabId = new JLabel("请输入实验室编号");

        jlNote.setPreferredSize(new Dimension(960, 30));
        jlUserId.setPreferredSize(new Dimension(160, 30));
        jlLabId.setPreferredSize(new Dimension(160, 30));
        jlNote.setText(note);

        buttonFaceDetection.setPreferredSize(new Dimension(260, 30));//设置大小
        buttonFaceCollectData.setPreferredSize(new Dimension(260, 30));
        buttonFaceTrain.setPreferredSize(new Dimension(260, 30));
        buttonOpenLab.setPreferredSize(new Dimension(260, 30));
        tfUserId.setPreferredSize(new Dimension(260, 30));;
        tfLabId.setPreferredSize(new Dimension(260, 30));;

        panel.add(jlNote);
        panel.add(jlUserId);
        panel.add(tfUserId);
        panel.add(buttonFaceDetection); //加到面板,不添加窗口中不显示
        panel.add(buttonFaceCollectData);
        
        panel.add(jlLabId);
        panel.add(tfLabId);
        panel.add(buttonFaceTrain);
        panel.add(buttonOpenLab);

        buttonFaceDetection.addActionListener(new ActionListener() {        //为按钮添加事件:开启人脸检测
            public void actionPerformed(ActionEvent e) {
                //输入按钮单单击后发生的动作
                startDetection();
            }
        });
        buttonFaceCollectData.addActionListener(new ActionListener() {        //为按钮添加事件:开启人脸收集
            public void actionPerformed(ActionEvent e) {
                //输入按钮单单击后发生的动作
                startCollectData(tfUserId.getText());
            }
        });
        buttonFaceTrain.addActionListener(new ActionListener() {        //为按钮添加事件:开启模型训练
            public void actionPerformed(ActionEvent e) {
                //输入按钮单单击后发生的动作
                startTrain(tfUserId.getText());
            }
        });
        buttonOpenLab.addActionListener(new ActionListener() {        //为按钮添加事件:开门方法
            public void actionPerformed(ActionEvent e) {
                //输入按钮单单击后发生的动作
                startOpenLab(tfUserId.getText(), tfLabId.getText());
//                fram.dispose();
            }
        });

        fram.add(panel);
        fram.setVisible(true);
    }

    private static <V, t> void startDetection() {
        System.out.println("-----------------startDetection-----------------");
//        FaceDetection faceDetection = new FaceDetection();
        new Thread(() -> {
            FaceDetection.main(null);
        }).start();
      
        
    }
    private static void startCollectData(String s) {
        System.out.println("-----------------startCollectData-----------------");
        System.out.println(s);
        CollectData.setId(s);
        new Thread(() -> {
    		try {
				CollectData.main(null);
			} catch (IOException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
			}
        }).start();
      
    }
    private static void startTrain(String s) {
        System.out.println("-----------------startTrain-----------------");
        Train.setId(s);
        new Thread(() -> {
        	try {
                Train.main(null);
            } catch (IOException e) {
                e.printStackTrace();
            }
        }).start();
        

    }
    private static void startOpenLab( String userId,String LabId) {
        System.out.println("-----------------startOpenLab-----------------");
        System.out.println("userId------->" + userId);
        System.out.println("LabId-------->" + LabId);
        
        Recognition.setUserId(userId);
        Recognition.setLabId(LabId);
        new Thread(() -> {
        	try {
				Recognition.main(null);
			} catch (Exception e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
			}
        }).start();
        
      
    }

}