代码改变世界

开源推荐 | 可实现门禁、AI测温敏捷开发的人脸识别应用套件

2020-07-07 16:24  码仔很忙  阅读(865)  评论(0编辑  收藏  举报

随着人脸识别技术的不断发展,尤其是在部分算法平台对外开放算法之后,人脸识别的应用门槛得到了极大降低。但是从算法到一款真正可落地的人脸识别产品,在完整的应用开发中,不仅需要考虑底层算法的运行逻辑,也需要上层业务逻辑完整自洽。而这个从0到1的过程,往往会让大部分中下企业及开发者,在项目商用化落地过程中无从下手。

因此,在这里推荐一款开源的人脸识别应用套件ArcFaceGo。基于该套件,可以快速甚至零代码搭建人脸识别闸机通行、刷脸考勤以及近来大火的人脸识别测温等应用。同时,利用它的开源属性,也完全能够在此基础上开发成一款符合自身业务需求的产品,并投入商用。

【简单易用 ArcFaceGo****应用套件构成】

在应用套件中,已经内置了虹软视觉开放平台的离线人脸识别SDK。涵盖人脸检测、人脸比对、活体检测、人脸属性分析等一系列核心算法,即使无网络状态下也可以运行。在虹软视觉开放平台完成下载后,可以看到该应用套件由人脸识别软件APK、中心管理端两部分构成。

中心管理端:拥有人员注册、考勤统计、出入查询等功能,也可以云端进一步对接已有的业务平台。

人脸识别设备端应用APK****:负责采集人脸信息、进行身份比对,并向中心管理端报备识别结果以及通知相关联通控制器。在该部分其代码中,已经针对人脸检测、人脸比对、活体检测等算法,模块化的集成了相应算法运转逻辑。开发者无需考虑各算法之间复杂运转流程,只需传入camera数据流,便可以回调识别结果。

同时,针对人脸图像传输、处理中需要用到的多媒体图像技术,在代码中也同样模块化的封装好了NV21、RGBA、BGR等不同颜色格式自动转化及图像的四字节对齐等功能。

在运行性能方面,该开源应用套件在底层算法逻辑中维护“人脸底库”,可以充分提升人脸识别速度,以RK3288 -10000人为例,相比在上层逻辑中进行人脸比对需要100ms ,而目前仅需10ms。

【适配灵活 可应用于各类实际场景****】

基于开源代码,开发者可以自行定义交互界面及交互逻辑,并开发应用于各类场景的人脸识别应用。比如智慧办公的人脸识别考勤机、智慧社区的人脸识别出入闸机、智慧商业的VIP迎宾系统、智慧医院的人脸识别挂号机、智慧酒店的人脸识别入住等等产品。

1.针对具体场景,可视化适配硬件

不同场景的不同应用,也意味着需要适配不同的分辨率设备,适配单摄、双摄、宽动态等各类摄像头。ArcFaceGo****人脸识别应用套件提供了可视化的硬件适配页面,方便预览显示及识别设置。

用户也能自行改写该部分代码,譬如不希望用户自行调节相关参数,以至于影响识别效果,甚至可以将该部分功能进行封装,以免用户误触。

2.可供对接硬件的多种广播形式。

Action 描述
com.arcsoft.arcfacesingle.ACTION_IDENTIFY_SUCCESSFUL 人员识别成功
com.arcsoft.arcfacesingle.ACTION_IDENTIFY_SUCCESS_PERSON_SERIAL 人员识别成功后,会向外发送“personSerial”唯一标识号”
com.arcsoft.arcfacesingle.ACTION_OPEN_DOOR 开门(在人员识别成功后会发送该广播)
com.arcsoft.arcfacesingle.ACTION_CLOSE_DOOR 关门(发送开门广播后,延迟一段时间会发送该广播)
com.arcsoft.arcfacesingle.ACTION_FACE_DETECT_HAS_FACE 检测到人脸框
com.arcsoft.arcfacesingle.ACTION_FACE_DETECT_NO_FACE 人脸框丢失

场景1

在室内走廊或楼梯过道等光线不佳的地方,可以对接ACTION_FACE_DETECT_HAS_FACE和ACTION_FACE_DETECT_NO_FACE广播,在检测到人脸时打开补光灯,这样可以提高识别率。

场景2

在一些单次授权通过的场景比如取药房,可对接ACTION_IDENTIFY_SUCCESS_PERSON_SERIAL广播,在授权并且识别通过之后可以删除该人授权以达到目的。

【人脸识别测温机开发实例】

疫情期间,很多智慧门禁设备都会增加测温模组,在通过人脸识别实现身份权限管理的同时,一并测量体温。

目前这类产品的市场反响非常火热,无论是要实现抗疫常态化的国内,还是感染人数持续暴增的国外,都对该产品有迫切需求。

这款应用套件可以与测温模块快速集成,以实现测温结果与识别身份绑定的效果。具体如何实现,以下是代码示例。

以下是具体实现的代码示例,以主流的HTPA32x32d为例。

private ConcurrentMap<Integer, Float> temperatureMap = new ConcurrentHashMap<>();

private byte[] rawData = new byte[2048];
private ReadThread.ReadListener readListener = (data, size) -> {synchronized (temperatureLocker) {
       System.arraycopy(data, 1, rawData, 0, 2048);
  }
};

public void start() {
   ReadThread readThread = new ReadThread();
readThread.setListener(readListener);
readThread.start();

FaceEngine faceEngine = new FaceEngine();
Config config = new Config();
   faceEngine.setRecognizeCallback(this);
faceEngine.init(this, config);
}
   
@Override
public void onRecognizeComplete(RecognitionResult recognitionResult) {
   FaceInfo faceInfo = recognitionResult.faceInfo;        
   if (recognitionResult.result == ErrorInfo.MOK) {
       if(temperatureMap.get(faceInfo.faceId)<=37.3) {
           //温度低于预警值并且识别成功,进行后续操作
      }
  }  
}

@Override
public void onCameraPreview(byte[] bytes, ICamera iCamera) {
FaceInfo faceInfo = new FaceInfo();
faceEngine.faceRecognize(rgbData, irData, width, height, true, faceInfo);

if (faceInfo.faceId != -1 && temperatureMap.get(faceInfo.faceId) != null) {
       //把额头区域映射至32x32温度矩阵中,取出对应的温度
       float temperature = getTemperature(faceInfo.forehead, rawData);
       temperatureMap.put(faceInfo.faceId, temperature);
}
}

public class ReadThread extends Thread {private static final String HEIMANN_PATH = "/dev/ttyS2";
   private static final int BAUDRATE = 115200;
private static final byte[] command = {(byte) 0XEE, (byte) 0xE1, (byte) 0x01, (byte) 0x55, (byte) 0xFF, (byte) 0xFC, (byte) 0xFD, (byte) 0xFF};    
   private FileOutputStream outputStream;
   private FileInputStream inputStream;
   private Object locker = new Object();
   
   public ReadThread() {
       try {mSerialPort = SerialPort
                  .newBuilder(HEIMANN_PATH, BAUDRATE)
                  .parity(0)
                  .dataBits(8)
                  .stopBits(1)
                  .build();
           mInputStream = (FileInputStream) mSerialPort.getInputStream();
           mOutputStream = (FileOutputStream) mSerialPort.getOutputStream();
      } catch (IOException e) {
           e.printStackTrace();
      }
  }
   
   public void sendCommand(byte[] data) {
       if (mOutputStream != null) {
           try {synchronized (locker) {
                   mOutputStream.write(data);
                   offset = 0;
              }
          } catch (IOException e) {
               e.printStackTrace();
          }
      }
  }
   
   @Override
   public void run() {
       super.run();
       while (!isInterrupted()) {
           int size;
           try {synchronized (locker) {
                   if (mInputStream == null) {
                       return;
                  }
                   size = mInputStream.available();
                   if (size > 0) {
                       mInputStream.read(rawData, offset, size);
                       offset += size;
                  }
              }

               if (offset == rawData.length) {
                   if (getListener() != null) {
                       getListener().onDataReceived(rawData, offset);
                  }
                   offset = 0;
              }
               
               if (offset == 0) {
                   SystemClock.sleep(50);
                   sendCommand(command);
              }
          } catch (IOException e) {
               e.printStackTrace();
          }
      }
  }
}

对该产品感兴趣的话可以到虹软视觉开放平台进一步了解哦~