iHRM 人力资源管理系统_第11章_刷脸登录

news2025/1/11 20:05:34

iHRM 人力资源管理系统_第11章_刷脸登录

文章目录

  • iHRM 人力资源管理系统_第11章_刷脸登录
  • 第11章 刷脸登录
  • 1 浅谈人工智能
    • 1.1 人工智能的概述
    • 1.2 人工智能的应用领域
    • 1.3 基于人工智能的刷脸登录介绍
  • 2 百度云AI概述
    • 2.1 概述
    • 2.2 百度云AI的开发步骤
    • 2.3 百度云AI的注册与认证
  • 3 百度云API的入门
    • 3.1 搭建环境
    • 3.2 人脸注册
    • 3.3 人脸更新
    • 3.4 人脸检测
    • 3.5 人脸查找
  • 4 刷脸登录实现
    • 4.1 需求分析
    • 4.2 搭建环境
    • 4.3 二维码生成
    • 4.4 封装API
    • 4.5 功能实现
      • 4.5.1 后端实现
      • 4.5.2 前端实现
        • (1) 二维码展现
        • (2) 落地页调用摄像头
    • 4.6 总结

第11章 刷脸登录

  • 理解刷脸登录的需求
  • 理解刷脸登录的开发流程
  • 实现刷脸登录功能

1 浅谈人工智能

1.1 人工智能的概述

人工智能(Artificial Intelligence),英文缩写为AI。它是研究、开发用于模拟、延伸和扩展人的智能的理论、方法、技术及应用系统的一门新的技术科学

人工智能是计算机科学的一个分支,它企图了解智能的实质,并生产出一种新的能以人类智能相似的方式做出反应 的智能机器,该领域的研究包括机器人、语言识别、图像识别、自然语言处理和专家系统等。人工智能从诞生以 来,理论和技术日益成熟,应用领域也不断扩大,可以设想,未来人工智能带来的科技产品,将会是人类智慧的“容器”。人工智能可以对人的意识、思维的信息过程的模拟。人工智能不是人的智能,但能像人那样思考、也可 能超过人的智能。

在这里插入图片描述

1.2 人工智能的应用领域

随着智能家电、穿戴设备、智能机器人等产物的出现和普及,人工智能技术已经进入到生活的各个领域,引发越来 越多的关注。

1.3 基于人工智能的刷脸登录介绍

刷脸登录是基于人工智能、生物识别、3D传感、大数据风控技术,最新实现的登录形式。用户在无需输入用户名 密码的前提下,凭借“刷脸”完成登录过程。实现刷脸登录的核心是人脸处理,在人脸处理中有两个概念:

  • 人脸检测:检测图中的人脸,并为人脸标记出边框。检测出人脸后,可对人脸进行分析,获得眼、口、鼻轮 廓等72个关键点定位准确识别多种人脸属性,如性别,年龄,表情等信息
  • 人脸识别(对比):通过提取人脸的特征,计算两张人脸的相似度,从而判断是否同一个人,并给出相似度 评分。

作为中小型企业,可以采取世面上流行的人工智能产品快速的实现刷脸登录需求。目前比较流行人脸检测产品如下

(我们的课程中使用百度云AI来完成人脸登录功能):

  • Face++
  • 腾讯优图
  • 科大讯飞
  • 百度云AI

2 百度云AI概述

2.1 概述

百度人脸识别基于深度学习的人脸识别方案,准确识别图片中的人脸信息,提供如下功能:

  • 人脸检测:精准定位图中人脸,获得眼、口、鼻等72个关键点位置,分析性别、年龄、表情等多种人脸属性
  • 人脸对比:对比两张人脸的相似度,并给出相似度评分,从而判断是否同一个人
  • 人脸搜索:针对一张人脸照片,在指定人脸集合中搜索,找出最相似的一张脸或多张人脸,并给出相似度分值
  • 活体检测:提供离线/在线方式的活体检测能力,判断操作用户是否为真人,有效抵御照片、视频、模具等作 弊攻击
  • 视频流人脸采集:设备端离线实时监测视频流中的人脸,同时支持处理静态图片或者视频流,输出人脸图片 并进行图片质量控制

2.2 百度云AI的开发步骤

    1. 注册账号创建应用
    1. 搭建工程导入依赖
    1. 人脸注册
    1. 人脸识别

2.3 百度云AI的注册与认证

(1) 注册百度云帐号

打开百度云平台:https://login.bce.baidu.com/reg.html?tpl=bceplat&from=portal进行账号注册

(2) 激活人脸识别,并创建应用

找到产品-人工智能-人脸识别激活应用,并注册应用

在这里插入图片描述

3 百度云API的入门

3.1 搭建环境

创建工程并导入依赖:

<dependency>
      <groupId>com.baidu.aip</groupId>
       <artifactId>java-sdk</artifactId>
       <version>4.8.0</version>
   </dependency>

3.2 人脸注册

用于从人脸库中新增用户,可以设定多个用户所在组,及组内用户的人脸图片

典型应用场景:构建您的人脸库,如会员人脸注册已有用户补全人脸信息等。

    //人脸注册
    @Test
    public void testFaceRegister() throws Exception {
        //传入可选参数调用接口
        HashMap<String, String> options = new HashMap<String, String>();
        options.put("quality_control", "NORMAL");
        options.put("liveness_control", "LOW");
        String imageType = "BASE64";
        String groupId = "itcast";
        String userId = "1000";


        //构造base64图片字符串
        String path = "C:\\Users\\ThinkPad\\Desktop\\ihrm\\day11\\资源\\照片\\001.png";
        byte[] bytes = Files.readAllBytes(Paths.get(path));
        String image = Base64Util.encode(bytes);
        // 人脸注册
        JSONObject res = client.addUser(image, imageType, groupId, userId, options);
        System.out.println(res.toString(2));
    }


人脸注册 请求参数详情

参数名称是否必选类型默认值说明
imageString图片信息(**总数据大小应小于10M),图片上传方式根据 image_type来判断
image_typeString图片类型 BASE64:图片的base64值,base64编码后的图片数据,需urlencode,编码后的图片大小不超过2M; URL:图片的 URL地址( 可能由于网络等原因导致下载图片时间过长)**;**FACE_TOKEN: 人脸图片的唯一标识,调用人脸检测接口时,会为每个人脸图片赋予一个唯一的FACE_TOKEN,同一张图片多次检测得到的FACE_TOKEN 是同一个
group_idString用户组id(由数字、字母、下划线组成),长度限制128B
user_idString用户id(由数字、字母、下划线组成),长度限制128B
user_infoString用户资料,长度限制256B
quality_controlStringNONE图片质量控制 NONE: 不进行控制 LOW:较低的质量要求NORMAL: 一般的质量要求 HIGH: 较高的质量要求 默认****NONE
liveness_controlStringNONE活体检测控制 NONE: 不进行控制 LOW:较低的活体要求 (高通过率 低攻击拒绝率) NORMAL: 一般的活体要求(平衡的攻击拒绝率, 通过率) HIGH: 较高的活体要求(高攻击拒绝率 低通过率) 默认****NONE

人脸注册 返回数据参数详情

字段必选类型说明
log_iduint64请求标识码,随机数,唯一
face_tokenstring人脸图片的唯一标识
locationarray人脸在图片中的位置
+leftdouble人脸区域离左边界的距离
+topdouble人脸区域离上边界的距离
+widthdouble人脸区域的宽度
+heightdouble人脸区域的高度
+rotationint64人脸框相对于竖直方向的顺时针旋转角,[-180,180]

3.3 人脸更新

用于对人脸库中指定用户,更新其下的人脸图像。

    //人脸更新
    @Test
    public void testFaceUpdate() throws Exception {
     //传入可选参数调用接口
        HashMap<String, String> options = new HashMap<String, String>();
        options.put("quality_control", "NORMAL");
        options.put("liveness_control", "LOW");
        String imageType = "BASE64";
        String groupId = "itcast";
        String userId = "1000";
       //构造base64图片字符串
        String path = "C:\\Users\\ThinkPad\\Desktop\\ihrm\\day11\\资源\\照片\\001.png";
        byte[] bytes = Files.readAllBytes(Paths.get(path));
        String image = Base64Util.encode(bytes);
      //人脸注册
        JSONObject res = client.updateUser(image, imageType, groupId, userId, options);
        System.out.println(res.toString(2));
    }


人脸更新 请求参数详情

参数名称是否必选类型默认****值说明
imageString图片信息(总数据大小应小于****10M),图片上传方式根据 image_type来判断
image_typeString图片类型 BASE64:图片的base64值,base64编码后的图片数据,需urlencode,编码后的图片大小不超过2M; URL:图片的 URL地址( 可能由于网络等原因导致下载图片时间过长);FACE_TOKEN: 人脸图片的唯一标识,调用人脸检测接口时,会为每个人脸图片赋予一个唯一的FACE_TOKEN,同一张图片多次检测得到的FACE_TOKEN 是同一个
group_idString更新指定groupid下uid对应的信息
user_idString用户id(由数字、字母、下划线组成),长度限制128B
user_infoString用户资料,长度限制256B
quality_controlStringNONE图片质量控制 NONE: 不进行控制 LOW:较低的质量要求NORMAL: 一般的质量要求 HIGH: 较高的质量要求 默认****NONE
liveness_controlStringNONE活体检测控制 NONE: 不进行控制 LOW:较低的活体要求 (高通过率 低攻击拒绝率) NORMAL: 一般的活体要求(平衡的攻击拒绝率, 通过率) HIGH: 较高的活体要求(高攻击拒绝率 低通过率) 默认****NONE

人脸更新 返回数据参数详情

字段必选类型说明
log_iduint64请求标识码,随机数,唯一
face_tokenstring人脸图片的唯一标识
locationarray人脸在图片中的位置
+leftdouble人脸区域离左边界的距离
+topdouble人脸区域离上边界的距离
+widthdouble人脸区域的宽度
+heightdouble人脸区域的高度
+rotationint64人脸框相对于竖直方向的顺时针旋转角,[-180,180]

3.4 人脸检测


    //人脸检测
    @Test
    public void testFaceDetect() throws IOException {
        String path = "C:\\Users\\ThinkPad\\Desktop\\ihrm\\day11\\资源\\照片\\002.png";
        byte[] bytes = Files.readAllBytes(Paths.get(path));
        String image = Base64Util.encode(bytes);
        String imageType = "BASE64";
        HashMap<String, String> subOptions = new HashMap<String, String>();
        subOptions.put("max_face_num", "10");
    //人脸检测
        JSONObject res = client.detect(image, imageType, subOptions);
        System.out.println(res.toString(2));
    }

人脸检测 请求参数详情

参数名称是否必选类型默认值说明
imageString图片信息(总数据大小应小于10M),图片上传方式根据image_type来判断
image_typeString图片类型 BASE64:图片的base64值,base64编码后的图片数据,需urlencode,编码后的图片大小不超过2M;URL:图片的 URL地址( 可能由于网络等原因导致下载图片时间过长)**;**FACE_TOKEN: 人脸图片的唯一标识,调用人脸检测接口时,会为每个人脸图片赋予一个唯一的FACE_TOKEN,同一张图片多次检测得到的FACE_TOKEN是同一个
face_fieldString包括**age,beauty,expression,faceshape,gender,glasses,landmark,race,quality,facetype信息逗号分隔. 默认只返回face_token、人脸框、概率和旋转角度
max_face_numString1最多处理人脸的数目,默认值为1,仅检测图片中面积最大的那个人脸;最大值****10,检测图片中面积最大的几张人脸。
face_typeString人脸的类型 LIVE表示生活照:通常为手机、相机拍摄的人像图片、或从网络获取的人像图片等IDCARD表示身份证芯片照:二代身份证内置芯片中的人像照片 WATERMARK表示带水印证件照:一般为带水印的小图,如公安网小图 CERT表示证件照片:如拍摄的身份证、工卡、护照、学生证等证件图片 默认LIVE

人脸检测 返回数据参数详情

字段必选类型说明
face_numint检测到的图片中的人脸数量
face_listarray人脸信息列表,具体包含的参数参考下面的列表。
+face_tokenstring人脸图片的唯一标识
+locationarray人脸在图片中的位置
++leftdouble人脸区域离左边界的距离
++topdouble人脸区域离上边界的距离
++widthdouble人脸区域的宽度
++heightdouble人脸区域的高度
++rotationint64人脸框相对于竖直方向的顺时针旋转角,[-180,180]
+face_probabilitydouble人脸置信度,范围【0~1】,代表这是一张人脸的概率,0最小、1 最大。
+angelarray人脸旋转角度参数
++yawdouble三维旋转之左右旋转角[-90(左), 90(右)]
++pitchdouble三维旋转之俯仰角度[-90(上), 90(下)]
++rolldouble平面内旋转角[-180(逆时针), 180(顺时针)]
+agedouble年龄 ,当face_field****包含age**时返回
+beautyint64美丑打分,范围0-100,越大表示越美。当face_fields包含beauty 时返回
+expressionarray表情,当 face_field****包含expression**时返回
++typestringnone:不笑;smile:微笑;laugh:大笑
++probabilitydouble表情置信度,范围【0~1】,0最小、1最大。
+face_shapearray脸型,当**face_field****包含faceshape时返回
++typedoublesquare: 正方形 triangle:三角形 oval: 椭圆 heart: 心形 round: 圆形
++probabilitydouble置信度,范围【0~1】,代表这是人脸形状判断正确的概率,0最小、1最大。
+genderarray性别,**face_field****包含gender时返回
++typestringmale:男性 female:女性
++probabilitydouble性别置信度,范围【0~1】,0代表概率最小、1代表最大。

3.5 人脸查找

在指定人脸集合中,找到最相似的人脸

 //人脸搜索
    @Test
    public void testFaceSearch() throws IOException {
        String path = "D:\\223.png";
        byte[] bytes = Files.readAllBytes(Paths.get(path));
        String image = Base64Util.encode(bytes);
        String imageType = "BASE64";
        HashMap<String, String> options = new HashMap<String, String>();
        options.put("user_top_num", "1");
    //人脸搜索
        JSONObject res = client.search(image, imageType, "itcast", options);
        System.out.println(res.toString(2));
    }

人脸搜索 请求参数详情

参数名称是否必选类型默认值说明
imageString图片信息(总数据大小应小于****10M),图片上传方式根据 image_type来判断
image_typeString图片类型 BASE64:图片的base64值,base64编码后的图片数据,需urlencode,编码后的图片大小不超过2M; URL:图片的 URL地址( 可能由于网络等原因导致下载图片时间过长)**;**FACE_TOKEN: 人脸图片的唯一标识,调用人脸检测接口时,会为每个人脸图片赋予一个唯一的FACE_TOKEN,同一张图片多次检测得到的FACE_TOKEN 是同一个
group_id_listString从指定的group中进行查找 用逗号分隔,上限20
quality_controlStringNONE图片质量控制 NONE: 不进行控制 LOW:较低的质量要求NORMAL: 一般的质量要求 HIGH: 较高的质量要求 默认****NONE
liveness_controlStringNONE活体检测控制 NONE: 不进行控制 LOW:较低的活体要求 (高通过率 低攻击拒绝率) NORMAL: 一般的活体要求(平衡的攻击拒绝率, 通过率) HIGH: 较高的活体要求(高攻击拒绝率 低通过率) 默认****NONE
user_idString当需要对特定用户进行比对时,指定user_id进行比对。即人脸认证功能。
max_user_numString查找后返回的用户数量。返回相似度最高的几个用户,默认为1,最多返回20个。

人脸搜索 返回数据参数详情

字段必选类型说明
face_tokenstring人脸标志
user_listarray匹配的用户信息列表
+group_idstring用户所属的group_id
+user_idstring用户的user_id
+user_infostring注册用户时携带的user_info
+scorefloat用户的匹配得分,推荐阈值80分

4 刷脸登录实现

4.1 需求分析

为了用户登录的便捷,我们在系统中增加刷脸登录的功能,大致流程如下图:

在这里插入图片描述

用户在登录页面触发刷脸登录功能

  • 该页面中弹出一个二维码,此二维码是后台即时生成,包含特殊标志(但本质上是一个URL链接),后续登 录流程将会使用此标志。用户对该二维码进行扫描,并在扫描端(手机或PC,注:此处不建议使用微信扫描)浏览器打开落地页。

  • 打开落地页时,授权使用摄像头,并进行人脸识别,识别成功后,关闭落地页。

  • 识别成功后,登录页面自动检测到成功标识,并获取相关信息,进入系统主页。

  • 技术点

    • 二维码生成
    • 百度云AI
    • Redis
    • 前端摄像头调用

4.2 搭建环境

(1) 引入坐标

  <dependency>
            <groupId>com.baidu.aip</groupId>
            <artifactId>java-sdk</artifactId>
            <version>4.8.0</version>
        </dependency>

        <!-- 百度云AI API-->
        <dependency>
            <groupId>com.baidu.aip</groupId>
            <artifactId>java-sdk</artifactId>
            <version>4.8.0</version>
        </dependency>
        <!-- 二维码 -->
        <dependency>
            <groupId>com.google.zxing</groupId>
            <artifactId>core</artifactId>
            <version>3.2.1</version>
        </dependency>
        <dependency>
        <groupId>com.google.zxing</groupId>
            <artifactId>javase</artifactId>
            <version>3.2.1</version>
        </dependency>

(2) 添加配置

ai:
  appId: 15191935
  apiKey: cyWSHgas93Vtdmt42OwbW8pu
  secretKey: yf1GusMvvLBdOnyubfLubNyod9iEDEZW
  imageType: BASE64
  groupId: itcast
qr:
  url: https://localhost:8080/#/facelogin

(3) 创建二维码工具类

配置二维码创建的工具类

    @Component
    public class QRCodeUtil {
        /**
         * 生成Base64 二维码
         */
        public String crateQRCode(String content) throws IOException {
            ByteArrayOutputStream os = new ByteArrayOutputStream();
            try {
                QRCodeWriter writer = new QRCodeWriter();
                BitMatrix bitMatrix = writer.encode(content, BarcodeFormat.QR_CODE, 200,
                        200);
                BufferedImage bufferedImage =
                        MatrixToImageWriter.toBufferedImage(bitMatrix);
                ImageIO.write(bufferedImage, "png", os);
//添加图片标识
                return new String("data:image/png;base64," +
                        Base64.encode(os.toByteArray()));
            } catch (Exception e) {
                e.printStackTrace();
            } finally {
                os.close();
            }
            return null;
        }
    }

在QRCodeUtil类头添加 @Component 注解,使用时可通过 @Autowired 来自动装配。

(4)创建基本的工程结构

在系统微服务中构建基本的Controller代码

@RestController
    @RequestMapping("/sys/faceLogin")
    public class FaceLoginController {
        /**
         * 获取刷脸登录二维码
         * 返回值:QRCode对象(code,image)
         *
         */
        @RequestMapping(value = "/qrcode", method = RequestMethod.GET)
        public Result qrcode() throws Exception {
            return null;
        }
        /**
         * 检查二维码:登录页面轮询调用此方法,根据唯一标识code判断用户登录情况
         * 查询二维码扫描状态
         * 返回值:FaceLoginResult
         * state :-1,0,1 (userId和token)
         */
        @RequestMapping(value = "/qrcode/{code}", method = RequestMethod.GET)
        public Result qrcodeCeck(@PathVariable(name = "code") String code) throws Exception
        {
            return null;
        }
        /**
         * 人脸登录:根据落地页随机拍摄的面部头像进行登录
         * 根据拍摄的图片调用百度云AI进行检索查找
         */
        @RequestMapping(value = "/{code}", method = RequestMethod.POST)
        public Result loginByFace(@PathVariable(name = "code") String code,
                                  @RequestParam(name = "file") MultipartFile attachment) throws Exception {
            return null;
        }
        /**
         * 图像检测,判断图片中是否存在面部头像
         */
        @RequestMapping(value = "/checkFace", method = RequestMethod.POST)
        public Result checkFace(@RequestParam(name = "file") MultipartFile attachment)
                throws Exception {
            return null;
        }
    }

在系统微服务中构建基本的Service代码

@Service
    public class FaceLoginService {
        @Value("${qr.url}")
        private String url;
        //创建二维码
        public QRCode getQRCode() throws Exception {
            return null;
        }
        //根据唯一标识,查询用户是否登录成功
        public FaceLoginResult checkQRCode(String code) {
            return null;
        }
        //扫描二维码之后,使用拍摄照片进行登录
        public String loginByFace(String code, MultipartFile attachment) throws Exception {
            return null;
        }
        //构造缓存key
        private String getCacheKey(String code) {
            return "qrcode_" + code;
        }
    }

4.3 二维码生成

    @Component
    public class QRCodeUtil {
        /**
         * 生成Base64 二维码
         */
        public String crateQRCode(String content) throws IOException {
            System.out.println(content);
            ByteArrayOutputStream os = new ByteArrayOutputStream();
            try {
                QRCodeWriter writer = new QRCodeWriter();
                BitMatrix bitMatrix = writer.encode(content, BarcodeFormat.QR_CODE, 200,
                        200);
                BufferedImage bufferedImage =
                        MatrixToImageWriter.toBufferedImage(bitMatrix);
                ImageIO.write(bufferedImage, "png", os);
//添加图片标识
                return new String("data:image/png;base64," +
                        Base64.encode(os.toByteArray()));
            } catch (Exception e) {
                e.printStackTrace();
            } finally {
                os.close();
            }
            return null;
        }
    }

在QRCodeUtil类头添加 @Component 注解,使用时可通过 @Autowired 来自动装配。

4.4 封装API

对于百度云AI SDK我们进行一些简单的封装,便于使用时,减少代码冗余。

  package com.ihrm.system.utils;
import com.baidu.aip.face.AipFace;
import org.json.JSONArray;
import org.json.JSONObject;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;
import javax.annotation.PostConstruct;
import java.util.HashMap;
    @Component
    public class BaiduAiUtil {
        @Value("${ai.appId}")
        private String APP_ID;
        @Value("${ai.apiKey}")
        private String API_KEY;
        @Value("${ai.secretKey}")
        private String SECRET_KEY;
        @Value("${ai.imageType}")
        private String IMAGE_TYPE;
        @Value("${ai.groupId}")
        private String groupId;
        private AipFace client;
        private HashMap<String, String> options = new HashMap<String, String>();

        public BaiduAiUtil() {
            options.put("quality_control", "NORMAL");
            options.put("liveness_control", "LOW");
        }

        @PostConstruct
        public void init() {
            client = new AipFace(APP_ID, API_KEY, SECRET_KEY);
        }
        /**
        *人脸注册 :将用户照片存入人脸库中
         */

        public Boolean faceRegister(String userId, String image) {
// 人脸注册
            JSONObject res = client.addUser(image, IMAGE_TYPE, groupId, userId, options);
            Integer errorCode = res.getInt("error_code");
            return errorCode == 0 ? true : false;
        }

        /**
         * 人脸更新 :更新人脸库中的用户照片
         */
        public Boolean faceUpdate(String userId, String image) {
// 人脸更新
            JSONObject res = client.updateUser(image, IMAGE_TYPE, groupId, userId,
                    options);
            Integer errorCode = res.getInt("error_code");
            return errorCode == 0 ? true : false;
        }

        /**
         * 人脸检测:判断上传图片中是否具有面部头像
         */
        public Boolean faceCheck(String image) {
            JSONObject res = client.detect(image, IMAGE_TYPE, options);
            if (res.has("error_code") && res.getInt("error_code") == 0) {
                JSONObject resultObject = res.getJSONObject("result");
                Integer faceNum = resultObject.getInt("face_num");
                return faceNum == 1 ? true : false;
            } else {
                return false;
            }
        }

        /**
         * 人脸查找:查找人脸库中最相似的人脸并返回数据
         * 处理:用户的匹配得分(score)大于80分,即可认为是同一个用户
         */
        public String faceSearch(String image) {
            JSONObject res = client.search(image, IMAGE_TYPE, groupId, options);
            if (res.has("error_code") && res.getInt("error_code") == 0) {
                JSONObject result = res.getJSONObject("result");
                JSONArray userList = result.getJSONArray("user_list");
                if (userList.length() > 0) {
                    JSONObject user = userList.getJSONObject(0);
                    double score = user.getDouble("score");
                    if (score > 80) {
                        return user.getString("user_id");
                    }
                }
            }
            return null;
        }
    }
  1. 在构造方法中,实例化client。通过client,可以调用SDK中包含的各种API。
  2. APP_ID, API_KEY, SECRET_KEY在文中第一段中所述位置获取,如没有正确配置,会直接导致API调用失败。
  3. 根据官方文档所示,我们大致创建了faceRegister()、faceUpdate()、faceCheck()、faceSearch()四个方法。
    • 人脸注册 faceRegister(groupId, userId, image)
    • groupId:用于人脸库区分人群标识,自定义即可,人脸库会根据提交的groupId,将用户分组
    • userId:人脸库中的用户标识,同组不可重复,自定义即可(通常为系统中用户的唯一标识)
    • image:Base64 用户图片
    • 人脸更新 faceUpdate(groupId, userId, image)
    • 参数解释同人脸注册
    • 该方法用于发生变化时,更新人脸信息
    • 人脸检测 faceCheck(image)
    • image:Base64 用户图片
    • 该方法用于人脸注册、人脸更新和人脸登录前使用
    • 目前采用的方案是检测出人脸数大于0即可,如需深化需求,可按需扩展
      人脸登录 faceSearch(image)
    • image:Base64 用户图片
    • 该方法使用的是百度云AI 人脸搜索方法,目前采用的方式是匹配度最高的结果,即要登录的用户

同样的,在BaiduAiUtil类头添加 @Component 注解,使用时可通过 @Autowired 来自动装配。在API调用后返回
值处理上,进行了简单的解析,如需深化解析,可按需扩展。

4.5 功能实现

完成刷脸登录一共需要我们解决如下5个问题:

人脸注册/人脸更新

  • 在刷脸登录之前,我们首先需要对系统中的用户进行人脸注册,将相关信息提交至人脸库,才可通过人脸识
  • 别的相关接口进行刷脸登录操作。当用户相貌变更较大时,可通过人脸更新进行人脸信息更换。

二维码生成

获取验证码。通过工具生成相关信息后,如特殊标志,将特殊标志写入Redis缓存,并将标记值设为”-1“,我 们认定值为”-1“,即为当前标记尚未使用。调用QRCodeUtil.crateQRCode()生成二维码。

二维码检测

前端获取二维码后,对二维码进行展现,并且前台启动定时器,定时检测特殊标记状态值。当状态值 为“1”时,表明登录成功。

人脸检测

当用户扫码进入落地页,通过落地页打开摄像头,并且定时成像。将成像图片,通过接口提交给后端进行人 脸检测。

人脸登录

检测成功后,即进行人脸登录,人脸登录后,改变特殊标记状态值,成功为“1”,失败为“0”。当登录成功时, 进行自动登录操作,将token和userId存入到redis中。

4.5.1 后端实现

(1)人脸注册/人脸更新:在刷脸登录之前,我们首先需要对系统中的用户进行人脸注册,将相关信息提交至人脸
库,才可通过人脸识别的相关接口进行刷脸登录操作。当用户相貌变更较大时,可通过人脸更新进行人脸信息更
换。

    //人脸注册
    @RequestMapping(value = "/register/face", method = RequestMethod.POST)
    public Boolean registerFace(@RequestParam(name = "fid") String fid) throws
            Exception {
        SysFile sysFile = fileService.findById(fid);
        String path = uploadPath + "/" + sysFile.getPath() + "/" +
                sysFile.getUuidName();
        byte[] bytes = Files.readAllBytes(Paths.get(path));
        Boolean isSuc;
        String image = Base64Utils.encodeToString(bytes);
        isSuc = userService.checkFace(image);
        if (isSuc) {
            isSuc = baiduAiUtil.faceRegister("1", userId, image);
        }
        return isSuc;
    }
    //人脸更新
    @RequestMapping(value = "/update/face", method = RequestMethod.POST)
    public boolean updateFace(@RequestParam(name = "fid") String fid) throws Exception
    {
        SysFile sysFile = fileService.findById(fid);
        String path = uploadPath + "/" + sysFile.getPath() + "/" +
                sysFile.getUuidName();
        byte[] bytes = Files.readAllBytes(Paths.get(path));
        Boolean isSuc;
        String image = Base64Utils.encodeToString(bytes);
        isSuc = userService.checkFace(image);
        if (isSuc) {
            isSuc = baiduAiUtil.faceUpdate("1", userId, image);
        }
        return isSuc;
    }

(2)二维码生成:获取验证码。通过工具生成相关信息后,如特殊标志,将特殊标志写入Redis缓存,并将标记值
设为”-1“,我们认定值为”-1“,即为当前标记尚未使用。调用QRCodeUtil.crateQRCode()生成二维码。

Controller

  /**
     * 获取刷脸登录二维码
     */
    @RequestMapping(value = "/qrcode", method = RequestMethod.GET)
    public Result qrcode() throws Exception {
        return new Result(ResultCode.SUCCESS, faceLoginService.getQRCode());
    }

Service

    public QRCode getQRCode() throws Exception {
        String code = idWorker.nextId() + "";
        FaceLoginResult result = new FaceLoginResult("-1");
        redisTemplate.boundValueOps(getCacheKey(code)).set(result, 30,
                TimeUnit.MINUTES);
        String strFile = qrCodeUtil.crateQRCode(url + "?code=" + code);
        return new QRCode(code, strFile);
    }

(3)二维码检测:前端获取二维码后,对二维码进行展现,并且前台启动定时器,定时检测特殊标记状态值。当 状态值为“1”时,表明登录成功。

Controller

  /**
     * 检查二维码:登录页面轮询调用此方法,根据唯一标识code判断用户登录情况
     */
    @RequestMapping(value = "/qrcode/{code}", method = RequestMethod.GET)
    public Result qrcodeCeck(@PathVariable(name = "code") String code) throws Exception
    {
        FaceLoginResult codeCheck = faceLoginService.checkQRCode(code);
        return new Result(ResultCode.SUCCESS, codeCheck);
    }

Service:

  public FaceLoginResult checkQRCode(String code) {
        String cacheKey = getCacheKey(code);
        FaceLoginResult result = (FaceLoginResult)
                redisTemplate.opsForValue().get(cacheKey);
        return result;
    }

(4)人脸检测/人脸登录:当用户扫码进入落地页,通过落地页打开摄像头,并且定时成像。将成像图片,通过接 口提交给后端进行人脸检测。

    /**
     * 图像检测,判断图片中是否存在面部头像
     */
    @RequestMapping(value = "/checkFace", method = RequestMethod.POST)
    public Result checkFace(@RequestParam(name = "file") MultipartFile attachment)
            throws Exception {
        if (attachment == null || attachment.isEmpty()) {
            throw new CommonException();
        }
        Boolean aBoolean =
                baiduAiUtil.faceCheck(Base64Utils.encodeToString(attachment.getBytes()));
        if(aBoolean) {
            return new Result(ResultCode.SUCCESS);
        }else{
            return new Result(ResultCode.FAIL);
        }
    }

(5)检测成功后,即进行人脸登录,人脸登录后,改变特殊标记状态值,成功为“1”,失败为“0”。当登录成功时, 进行自动登录操作,将token和userId存入到redis中。

Controller

    @RequestMapping(value = "/{code}", method = RequestMethod.POST)
    public Result loginByFace(@PathVariable(name = "code") String code,
                              @RequestParam(name = "file") MultipartFile attachment) throws Exception {
        String userId = faceLoginService.loginByFace(code, attachment);
        if(userId == null) {
            return new Result(ResultCode.FAIL);
        }else{
//构造返回数据
            return new Result(ResultCode.SUCCESS);
        }
    }

Service


    public String loginByFace(String code, MultipartFile attachment) throws Exception {
        String userId =
                baiduAiUtil.faceSearch(Base64Utils.encodeToString(attachment.getBytes()));
        FaceLoginResult result = new FaceLoginResult("1");
        if(userId != null) {
            User user = userDao.findById(userId).get();
            if(user != null) {
                Subject subject = SecurityUtils.getSubject();
                subject.login(new UsernamePasswordToken(user.getMobile(),
                        user.getPassword()));
                String token = subject.getSession().getId() + "";
                result = new FaceLoginResult("0",token,userId);
            }
        }
        redisTemplate.boundValueOps(getCacheKey(code)).set(result, 30,
                TimeUnit.MINUTES);
        return userId;
    }

4.5.2 前端实现

前端主要实现的功能是,获取二维码并展示,然后后台轮询检测刷脸登录状态,并且实现落地页相关功能(摄像头 调用、定时成像、发送人脸检测和发送人脸登录请求)

(1) 二维码展现

 // 二维码
    handlecode() {
        qrcode().then(res => {

                this.param.qrcode = res.data.file
                this.centerDialogVisible = true
                this.codeCheckInfo = res.data.code
                setInterval(() => {
        if (this.states === '-1') {
            codeCheck({ code: res.data.code }).then(res => {
                    this.states = res.data.state
                    this.token = res.data.token
            if (this.states === '0') {
                     // 登录
                this.$store
                        .dispatch('LoginByCode', res.data.token)
                        .then(() => {
                        this.$router.push({ path: '/' })
                })
                 .catch(() => {
                })
            }
            if (this.states === '1') {
                 // 关闭
                this.centerDialogVisible = false
            }
          })
        }
      }, 1000 * 10)
   })
  }

(2) 落地页调用摄像头

    handleClick() {
        let _this = this


        if (!this.vdstate) { return false
        }
        if (!_this.states) {
// 注册拍照按钮的单击事件
            let video = this.$refs['vd'] let canvas = this.$refs['cav']
// let form = this.$refs["myForm"];
            let context = canvas.getContext('2d')
// 绘制画面
            context.drawImage(video, 0, 0, 200, 200)
            let base64Data = canvas.toDataURL('image/jpg')

// 封装blob对象
            let blob = this.dataURItoBlob(base64Data, 'camera.jpg') // base64 转图片file
            let formData = new FormData() formData.append('file', blob)

            this.imgUrl = base64Data
            checkFace(formData).then(res => {
            if (res.data.isSuc) {
                axios({
                        method: 'post',
                        url: '/api/frame/facelogin/' + this.$route.query.code,
                        data: formData
})
.then(function(response) {
                    console.log(response)
                    _this.states = true
                    _this.canvasShow = false
                    _this.tipShow = true
// _this.$message.success('验证通过' + '!')
                })
.catch(function(error) {
                    console.log(error)
                })
            } else {
                return false
            }
})
        }
    },
    dataURItoBlob(base64Data) {
        var byteString
        if (base64Data.split(',')[0].indexOf('base64') >= 0)
            byteString = atob(base64Data.split(',')[1])
        else byteString = unescape(base64Data.split(',')[1])
        var mimeString = base64Data
                .split(',')[0]
                .split(':')[1]
                .split(';')[0]
        var ia = new Uint8Array(byteString.length)
        for (var i = 0; i < byteString.length; i++) {
            ia[i] = byteString.charCodeAt(i)
        }
        return new Blob([ia], { type: mimeString })
    }
}
     

4.6 总结

= false
_this.tipShow = true
// _this.$message.success(‘验证通过’ + ‘!’)
})
.catch(function(error) {
console.log(error)
})
} else {
return false
}
})
}
},
dataURItoBlob(base64Data) {
var byteString
if (base64Data.split(‘,’)[0].indexOf(‘base64’) >= 0)
byteString = atob(base64Data.split(‘,’)[1])
else byteString = unescape(base64Data.split(‘,’)[1])
var mimeString = base64Data
.split(‘,’)[0]
.split(‘:’)[1]
.split(‘;’)[0]
var ia = new Uint8Array(byteString.length)
for (var i = 0; i < byteString.length; i++) {
ia[i] = byteString.charCodeAt(i)
}
return new Blob([ia], { type: mimeString })
}
}


## 4.6	总结
通过上述的步骤,可以实现一个刷脸登录的功能,其核心在于百度云AI的使用。通过合理的使用百度云AI SDK提供的相关API,我们可以很轻松的实现刷脸登录功能。刷脸登录的业务流程有很多种,我们只是实现了一种借助二维 码的方式,作为抛砖引玉。更多的流程和实现方式,在此不进行赘述。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/19829.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

web大作业 静态网页(地下城与勇士 10页 带视频)

⛵ 源码获取 文末联系 ✈ Web前端开发技术 描述 网页设计题材&#xff0c;DIVCSS 布局制作,HTMLCSS网页设计期末课程大作业 | 游戏官网 | 游戏网站 | 电竞游戏 | 游戏介绍 | 等网站的设计与制作 | HTML期末大学生网页设计作业&#xff0c;Web大学生网页 HTML&#xff1a;结构 …

开源免费的对象存储Minio

什么是Minio&#xff1f; Minio是一个基于Apache License v2.0开源协议的对象存储服务。它兼容亚马逊S3云存储服务接口&#xff0c;非常适合存储大容量、非结构化的数据。例如&#xff0c;图片、视频、日志文件、备份数据和容器/虚拟机镜像等&#xff0c;而一个对象文件可以是…

机器学习:卷积神经网络

卷积神经网络卷积神经网络的结构及原理卷积层池化层激活函数全连接层反馈运算使用MNIST数据集进行代码解析数据介绍实现流程代码实现卷积神经网络的结构及原理 卷积层 卷积运算一个重要的特点就是&#xff1a;通过卷积运算&#xff0c;可以使原信号特征增强&#xff0c;并且降…

服务器常用的异常及性能排查

服务器常用的异常及性能排查 使用 top 命令查看性能指标 top 命令使用详细介绍&#xff1a;传送门 查看Tasks total 进程数 正常我们在使用过程中对每天的一个进程数大概是有一个谱的&#xff0c;比如正常就是1百多个&#xff0c;突然暴增几百&#xff0c;那就很明显这里有…

计算机网络:运输层

运输层 运输层主要解决了应用进程之间的通信&#xff0c;称之为端到端协议 1.运输层概述 计算机网-------络体系结构的角度 AP:应用进程之间的简称 2. 运输层端口号、复用与分用的概念 2.1 端口号 2.2 发送方的复用&#xff08;multiplexing&#xff09;和接收方的分用&…

【Java面试八股文宝典之基础篇】备战2023 查缺补漏 你越早准备 越早成功!!!——Day10

大家好&#xff0c;我是陶然同学&#xff0c;软件工程大三明年实习。认识我的朋友们知道&#xff0c;我是科班出身&#xff0c;学的还行&#xff0c;但是对面试掌握不够&#xff0c;所以我将用这100多天更新Java面试题&#x1f643;&#x1f643;。 不敢苟同&#xff0c;相信大…

锐捷RLDP理论及实验讲解

RLDP概念 RLDP&#xff08;Rapid Link Detection Protocol&#xff09;是一个用于快速检测以太网链路故障的链路协议&#xff0c;包括环路链路故障、单向链路故障、双向链路故障等 工作原理 RLDP定义了两种协议报文&#xff1a;探测报文&#xff08;Probe&#xff09;和探测响…

【Java第32期】:Spring 中普通Maven项目的创建

作者&#xff1a;有只小猪飞走啦 博客地址&#xff1a;https://blog.csdn.net/m0_62262008?typeblog 内容&#xff1a;Spring 中普通Maven项目的创建 文章目录前言一&#xff0c;创建Spring项目1.创建一个普通的Maven项目2&#xff0c;添加Spring框架3&#xff0c;添加启动类…

【数据库系统概论】关系数据理论、范式

数据库一二三范式简单解释 第一范式 一个关系模式应当是一个五元组。 R(U,D,DOM,F)R(U,D,DOM,F)R(U,D,DOM,F) 这里&#xff1a; 关系名RRR是符号化的元组语义UUU为一组属性DDD为属性组UUU中的属性所来自的域DOMDOMDOM为属性到域的映射FFF为属性组UUU上的一组数据依赖 由于D…

RabbitMQ_概述

RabbitMQ大致工作流程图 解释 Producer&#xff1a;生产者 Consumer&#xff1a;消费者 Connection&#xff1a;AMQP协议连接 Channel&#xff1a;信道&#xff0c;进行消息读写的通道&#xff0c;RabbitMQ的绝大部分操作在信道完成&#xff1b;客户端可以建立多个信道&…

用 AWTK 和 AWPLC 快速开发嵌入式应用程序 (4)- 自定义功能块(上)

AWPLC 目前还处于开发阶段的早期&#xff0c;写这个系列文章的目的&#xff0c;除了用来验证目前所做的工作外&#xff0c;还希望得到大家的指点和反馈。如果您有任何疑问和建议&#xff0c;请在评论区留言。 1. 背景 AWTK 全称 Toolkit AnyWhere&#xff0c;是 ZLG 开发的开源…

PTA题目 两个数的简单计算器

本题要求编写一个简单计算器程序&#xff0c;可根据输入的运算符&#xff0c;对2个整数进行加、减、乘、除或求余运算。题目保证输入和输出均不超过整型范围。 输入格式&#xff1a; 输入在一行中依次输入操作数1、运算符、操作数2&#xff0c;其间以1个空格分隔。操作数的数…

跟艾文学编程《Python基础》(7)pandas数据分析

作者&#xff1a; 艾文&#xff0c;计算机硕士学位&#xff0c;企业内训讲师和金牌面试官&#xff0c;公司资深算法专家&#xff0c;现就职BAT一线大厂。邮箱&#xff1a; 1121025745qq.com博客&#xff1a;https://wenjie.blog.csdn.net/内容&#xff1a;跟艾文学编程《Python…

汉字风格迁移篇---W-net:基于深度神经网络的一次任意风格汉字生成

文章目录一、摘要二、提出原因已有的一些模型解决方案依然存在的限制三、介绍与创新四、模型介绍预处理w-net结构优化策略和损失函数五、实验实验设置用zi2zi作为基线具体实现1、 W-Net训练期间的超参数设置如下&#xff1a;2、一些细节处理模型评估W-net、zi2zi-v1、zi2zi-v2不…

第2-3-7章 个人网盘服务接口开发-文件存储服务系统-nginx/fastDFS/minio/阿里云oss/七牛云oss

文章目录5.8 导入其他接口代码5.8.1 接口导入-分页查询附件5.8.2 接口导入-根据业务类型/业务id查询附件5.9 导入网盘服务接口5.9.1 导入FileController5.9.2 导入StatisticsController5.9.3 导入FileRestManager5.9.4 导入FileService5.9.5 导入FileServiceImpl5.9.6 扩展File…

面向OLAP的列式存储DBMS-8-[ClickHouse]的常用聚合函数

ClickHouse 中的常用聚合函数 1 聚合函数 ClickHouse 中的聚合函数&#xff0c;因为和关系型数据库的相似性&#xff0c;本来聚合函数不打算说的&#xff0c;但是 ClickHouse 提供了很多关系型数据库中没有的函数&#xff0c;所以我们还是从头了解一下。 1.1 count count&…

Vue3 用src动态引入本地图片

&#x1f4ad;&#x1f4ad; ✨&#xff1a; Vue3 用src动态引入本地图片   &#x1f49f;&#xff1a;东非不开森的主页   &#x1f49c;: 躲起来的星星也在努力发光 你也要&#x1f49c;&#x1f49c;   &#x1f338;: 如有错误或不足之处&#xff0c;希望可以指正&#…

Qt OpenGL(二十二)——Qt OpenGL 核心模式-VAO和VBO

Qt OpenGL(二十二)——Qt OpenGL 核心模式-VAO和VBO 一、再谈VAO、VBO 上一篇文章,通过VAO、VBO绘制了一个三角形,过程需要创建VAO、VBO和释放。之所以有这些步骤,就是因为OpenGL本质就是一个大的状态机。但是我们如果要继续学习核心模式的OpenGL的话,VAO、VBO是我们必…

Java集合(一):泛型与Collection集合

目录 集合预热&#xff1a;泛型 泛型的优点 自定义泛型类型 自定义泛型类/接口 泛型使用细节 自定义泛型方法 泛型与继承关系 不存在继承关系的情况 通配符与存在继承关系的情况 泛型受限 集合概述 集合的作用与存储内容 集合与数据结构 集合&#xff1a;Collectio…

【基础算法系列】离散化与前缀和算法的运用

⭐️前面的话⭐️ 本篇文章将主要介绍离散化算法&#xff0c;所谓离散化算法&#xff0c;就是将一个无限区间上散点的数&#xff0c;在不改变相对大小的情况下&#xff0c;映射到一个较小的区间当中&#xff0c;然后对这个较小的区间进行操作的过程就是离散化的过程&#xff0…