聪明的人脸识别4——Pytorch 利用Retinaface+Facenet搭建人脸识别平台

article/2025/8/28 0:24:35

睿智的目标检测51——Pytorch 利用Retinaface+Facenet搭建人脸识别平台

  • 学习前言
  • 什么是Retinface和Facenet
    • 1、Retinface
    • 2、Facenet
  • 整体实现代码
  • 实现流程
    • 一、数据库的初始化
    • 二、检测图片的处理
      • 1、人脸的截取与对齐
      • 2、利用Facenet对矫正后的人脸进行编码
      • 3、将实时图片中的人脸特征与数据库中的进行比对
      • 4、图片绘制
  • 使用Retinaface+Facenet进行人脸识别:

学习前言

我又死了我又死了我又死了!
在这里插入图片描述

什么是Retinface和Facenet

1、Retinface

Retinaface是来自insightFace的又一力作,基于one-stage的人脸检测网络。
同时开源了代码与数据集,在widerface上有非常好的表现。
在这里插入图片描述

2、Facenet

谷歌人脸识别算法,发表于 CVPR 2015,利用相同人脸在不同角度等姿态的照片下有高内聚性,不同人脸有低耦合性,提出使用 cnn + triplet mining 方法,在 LFW 数据集上准确度达到 99.63%。

通过 CNN 将人脸映射到欧式空间的特征向量上,实质上:不同图片人脸特征的距离较大;通过相同个体的人脸的距离,总是小于不同个体的人脸这一先验知识训练网络。

测试时只需要计算人脸特征EMBEDDING,然后计算距离使用阈值即可判定两张人脸照片是否属于相同的个体。
在这里插入图片描述
简单来讲,在使用阶段,facenet即是:
1、输入一张人脸图片
2、通过深度卷积网络提取特征
3、L2标准化
4、得到一个长度为128特征向量。

整体实现代码

https://github.com/bubbliiiing/facenet-retinaface-pytorch

Retinaface原理和Facenet原理可以参考我的另外两篇博客。
睿智的目标检测42——Pytorch搭建Retinaface人脸检测与关键点定位平台

聪明的人脸识别3——Pytorch 搭建自己的Facenet人脸识别平台

实现流程

整体的代码摆放如下:
在这里插入图片描述

一、数据库的初始化

在这里插入图片描述
face_dataset里面装的是想要识别的人脸,在图片中看到的obama_1.jpg指的就是obama的第一张人脸图片,可以配置多张图片都指向obama,如obama_2.jpg、obama_3.jpg,需要注意的是,face_dataset里面每张图片都只能包含一张人脸,即目标人脸。

数据库中每一张图片对应一个人的人脸,图片名字中“_”靠左的部分就是这个人的名字。

数据库初始化指的是人脸数据库的初始化

想要实现人脸识别,首先要知道自己需要识别哪些人脸,在这一步中,我们会讲识到的人脸进行编码并放入数据库中。

数据库的初始化具体执行的过程就是

1、遍历数据库中所有的图片。
2、利用Retinaface检测每个图片中的人脸位置。
3、将人脸截取下来。
4、将获取到的人脸进行对齐。
5、利用Facenet将人脸进行编码。
6、将所有人脸编码的结果放在一个列表中。
7、保存成npy的形式。

第6步得到的列表就是已知的所有人脸的特征列表,在之后获得的实时图片中的人脸都需要与已知人脸进行比对,这样我们才能知道谁是谁
实现代码如下:

import osfrom retinaface import Retinaface'''
在更换facenet网络后一定要重新进行人脸编码,运行encoding.py。
'''
retinaface = Retinaface(1)list_dir = os.listdir("face_dataset")
image_paths = []
names = []
for name in list_dir:image_paths.append("face_dataset/"+name)names.append(name.split("_")[0])retinaface.encode_face_dataset(image_paths,names)

二、检测图片的处理

1、人脸的截取与对齐

在这里插入图片描述

利用Retinaface我们可以获得一张图片中人脸的位置,但是我们截取下来的人脸是这样的:
在这里插入图片描述
我们可以很明显的看出来人脸是歪着的,我们如果人脸可以正过来,那么将对人脸的特征提取非常有好处。
在这里插入图片描述
下面这张图看着就正多了。

常见的对齐方法有很多,在本篇博客里我们使用双眼坐标进行旋正。

利用双眼坐标进行旋正需要用到两个参数,如图所示分别是:
1、眼睛连线相对于水平线的倾斜角。
2、图片的中心。

在这里插入图片描述
利用这两个参数我们可以知道需要图片需要旋转的角度是多少,图片旋转的中心是什么

代码实现如下,其中landmark是五个人脸特征点的位置:

#-------------------------------------#
#   人脸对齐
#-------------------------------------#
def Alignment_1(img,landmark):if landmark.shape[0]==68:x = landmark[36,0] - landmark[45,0]y = landmark[36,1] - landmark[45,1]elif landmark.shape[0]==5:x = landmark[0,0] - landmark[1,0]y = landmark[0,1] - landmark[1,1]# 眼睛连线相对于水平线的倾斜角if x==0:angle = 0else: # 计算它的弧度制angle = math.atan(y/x)*180/math.picenter = (img.shape[1]//2, img.shape[0]//2)RotationMatrix = cv2.getRotationMatrix2D(center, angle, 1)# 仿射函数new_img = cv2.warpAffine(img,RotationMatrix,(img.shape[1],img.shape[0])) RotationMatrix = np.array(RotationMatrix)new_landmark = []for i in range(landmark.shape[0]):pts = []    pts.append(RotationMatrix[0,0]*landmark[i,0]+RotationMatrix[0,1]*landmark[i,1]+RotationMatrix[0,2])pts.append(RotationMatrix[1,0]*landmark[i,0]+RotationMatrix[1,1]*landmark[i,1]+RotationMatrix[1,2])new_landmark.append(pts)new_landmark = np.array(new_landmark)return new_img, new_landmark#---------------------------------------------------#
#   检测图片
#---------------------------------------------------#
def detect_image(self, image):# 绘制人脸框image = np.array(image, np.float32)old_image = np.array(image.copy(), np.uint8)#---------------------------------------------------##   Retinaface检测部分-开始#---------------------------------------------------#im_height, im_width, _ = np.shape(image)# 它的作用是将归一化后的框坐标转换成原图的大小scale = torch.Tensor([np.shape(image)[1], np.shape(image)[0], np.shape(image)[1], np.shape(image)[0]])scale_for_landmarks = torch.Tensor([np.shape(image)[1], np.shape(image)[0], np.shape(image)[1], np.shape(image)[0],np.shape(image)[1], np.shape(image)[0], np.shape(image)[1], np.shape(image)[0],np.shape(image)[1], np.shape(image)[0]])if self.letterbox_image:image = letterbox_image(image,[self.cfg["image_size"], self.cfg["image_size"]])anchors = Anchors(self.cfg, image_size=(self.cfg["image_size"], self.cfg["image_size"])).get_anchors()else:anchors = Anchors(self.cfg, image_size=(im_height, im_width)).get_anchors()# pytorchimage = preprocess_input(image).transpose(2, 0, 1)# 增加batch_size维度image = torch.from_numpy(image).unsqueeze(0).type(torch.FloatTensor)if self.cuda:scale = scale.cuda()scale_for_landmarks = scale_for_landmarks.cuda()image = image.cuda()anchors = anchors.cuda()with torch.no_grad():loc, conf, landms = self.net(image)  # forward passboxes = decode(loc.data.squeeze(0), anchors, self.cfg['variance'])boxes = boxes * scaleboxes = boxes.cpu().numpy()conf = conf.data.squeeze(0)[:,1:2].cpu().numpy()landms = decode_landm(landms.data.squeeze(0), anchors, self.cfg['variance'])landms = landms * scale_for_landmarkslandms = landms.cpu().numpy()boxes_conf_landms = np.concatenate([boxes,conf,landms],-1)boxes_conf_landms = non_max_suppression(boxes_conf_landms, self.confidence)if len(boxes_conf_landms)<=0:return old_imageboxes_conf_landms = np.array(boxes_conf_landms)if self.letterbox_image:boxes_conf_landms = retinaface_correct_boxes(boxes_conf_landms, np.array([self.cfg["image_size"], self.cfg["image_size"]]), np.array([im_height, im_width]))#---------------------------------------------------##   Retinaface检测部分-结束#---------------------------------------------------#

2、利用Facenet对矫正后的人脸进行编码

在这里插入图片描述

Facenet是一个人脸特征获取的模型,将第1步获得的对齐人脸传入Facenet模型就可以得到每个人脸的特征向量。

将所有特征向量保存在一个列表中,在第3步进行比对。

#-----------------------------------------------#
#   Facenet编码部分-开始
#-----------------------------------------------#
face_encodings = []
for boxes_conf_landm in boxes_conf_landms:#----------------------##   图像截取,人脸矫正#----------------------#crop_img = np.array(old_image)[int(boxes_conf_landm[1]):int(boxes_conf_landm[3]), int(boxes_conf_landm[0]):int(boxes_conf_landm[2])]landmark = np.reshape(boxes_conf_landm[5:],(5,2)) - np.array([int(boxes_conf_landm[0]),int(boxes_conf_landm[1])])crop_img,_ = Alignment_1(crop_img,landmark)#----------------------##   人脸编码#----------------------#crop_img = np.array(letterbox_image(np.uint8(crop_img),(self.input_shape[1],self.input_shape[0])))/255crop_img = np.expand_dims(crop_img.transpose(2, 0, 1),0)with torch.no_grad():crop_img = torch.from_numpy(crop_img).type(torch.FloatTensor)if self.cuda:crop_img = crop_img.cuda()# 利用facenet_model计算长度为128特征向量face_encoding = self.facenet(crop_img)[0].cpu().numpy()face_encodings.append(face_encoding)
#-----------------------------------------------#
#   Facenet编码部分-结束
#-----------------------------------------------#

3、将实时图片中的人脸特征与数据库中的进行比对

在这里插入图片描述

这个比对过程需要循环实现,具体对实时图片中的每一个人脸进行循环:
1、获取实时图片中的每一个人脸特征。
2、将每一个人脸特征和数据库中所有的人脸进行比较,计算距离。如果距离小于门限值,则认为其具有一定的相似度。
3、获得每一张人脸在数据库中最相似的人脸的序号。
4、判断这个序号对应的人脸距离是否小于门限,是则认为人脸识别成功,他就是这个人。

实现代码如下:


#-----------------------------------------------#
#   人脸特征比对-开始
#-----------------------------------------------#
face_names = []
for face_encoding in face_encodings:# 取出一张脸并与数据库中所有的人脸进行对比,计算得分matches, face_distances = compare_faces(self.known_face_encodings, face_encoding, tolerance = self.facenet_threhold)name = "Unknown"# 取出这个最近人脸的评分best_match_index = np.argmin(face_distances)if matches[best_match_index]:name = self.known_face_names[best_match_index]face_names.append(name)
#-----------------------------------------------#
#   人脸特征比对-结束
#-----------------------------------------------#

4、图片绘制

这一部分只是检测结果绘制在图片上,由绘制代码组成。

for i, b in enumerate(results):text = "{:.4f}".format(b[4])b = list(map(int, b))cv2.rectangle(old_image, (b[0], b[1]), (b[2], b[3]), (0, 0, 255), 2)cx = b[0]cy = b[1] + 12cv2.putText(old_image, text, (cx, cy),cv2.FONT_HERSHEY_DUPLEX, 0.5, (255, 255, 255))# landmscv2.circle(old_image, (b[5], b[6]), 1, (0, 0, 255), 4)cv2.circle(old_image, (b[7], b[8]), 1, (0, 255, 255), 4)cv2.circle(old_image, (b[9], b[10]), 1, (255, 0, 255), 4)cv2.circle(old_image, (b[11], b[12]), 1, (0, 255, 0), 4)cv2.circle(old_image, (b[13], b[14]), 1, (255, 0, 0), 4)name = face_names[i]font = cv2.FONT_HERSHEY_SIMPLEXcv2.putText(old_image, name, (b[0] , b[3] - 15), font, 0.75, (255, 255, 255), 2) 

使用Retinaface+Facenet进行人脸识别:

在GITHUB上下载好库后将库解压。
在这里插入图片描述
下载对应的权重,README中会有下载连接,百度网盘下载或者GITHUB下载均可。
在这里插入图片描述
在retinaface.py里面,根据自身需求修改retinaface_model_path、retinaface_backbone、facenet_model_path、facenet_backbone四个参数。
在这里插入图片描述
运行encoding.py进行人脸数据集编码。
在这里插入图片描述
运行predict.py进行人脸图片的预测。
在这里插入图片描述


http://chatgpt.dhexx.cn/article/LG5Smmo4.shtml

相关文章

Facenet 原理介绍

引子[编辑 | 编辑源代码] 这篇wiki主要介绍facenet人脸相似比较的基本原理&#xff0c;另外两篇wiki主要介绍基于tensorflow实现facenet的准确率测试及源码解读。经过在网上的一番搜索&#xff0c;找到了facenet实现人脸聚类的论文和论文解读&#xff0c;以及github上根据facen…

【 facenet-retinaface】快速复现 实现 facenet-retinaface-pytorch 人脸识别 windows上 使用cpu实现

目录 0 前言1 搭建环境与项目2 人脸预测与结果展示 0 前言 这一次要复现的是人脸识别中的 facenet-retinaface-pytorch 是在上一次博客的内容上更进一步 快速复现 实现 facenet-pytorch 人脸识别 windows上 使用cpu实现 人脸对比 参考了&#xff1a; Pytorch 利用Facenet和Re…

FaceNet

摘要&#xff1a; 尽管人脸识别领域最近取得了重大进展[10,14,15,17]&#xff0c;但大规模有效地实施人脸验证和识别对当前方法提出了严峻挑战。在本文中&#xff0c;我们提出了一个称为 FaceNet 的系统&#xff0c;它直接学习从人脸图像到紧凑欧几里得空间的映射&#xff0c;其…

【facenet】快速复现 实现 facenet-pytorch 人脸识别 windows上 使用cpu实现 人脸对比

目录 0 前言1 搭建环境与项目2 人脸预测与结果展示 0 前言 这一次要复现的是人脸识别中的 facenet-pytorch 参考了&#xff1a; Pytorch 搭建自己的Facenet人脸识别网络&#xff08;Bubbliiiing 深度学习 教程&#xff09; https://gitee.com/xiaozhao123666/facenet-pytorch …

利用MTCNN和facenet实现人脸检测和人脸识别

利用MTCNN和facenet实现人脸检测和人脸识别 人脸检测和人脸识别技术算是目前人工智能方面应用最成熟的技术了。本博客将利用mtcnn和faceNet搭建一个实现人脸检测和人脸识别的系统。基本思路也很简单&#xff0c;先利用mtcnn的进行人脸检测&#xff0c;当然也可以使用其他的人脸…

facenet 总结一

Facenet是谷歌研发的人脸识别系统&#xff0c;该系统是基于百万级人脸数据训练的深度卷积神经网络&#xff0c;可以将人脸图像embedding&#xff08;映射&#xff09;成128维度的特征向量。以该向量为特征&#xff0c;采用knn或者svm等机器学习方法实现人脸识别。 CASIA-WebFac…

【人脸识别】FaceNet详解

论文题目&#xff1a;《FaceNet: A Unified Embedding for Face Recognition and Clustering》 论文地址&#xff1a;FaceNet 1、概述 FaceNet&#xff08;A Unified Embedding for Face Recognition and Clustering&#xff09;直接把输入图像变成欧式空间中的特征向量&#…

人脸识别系统FaceNet原理

1. 概述 近年来&#xff0c;随着深度学习在CV领域的广泛应用&#xff0c;人脸识别领域也得到了巨大的发展。在深度学习中&#xff0c;通过多层网络的连接&#xff0c;能够学习到图像的特征表示&#xff0c;那么两张人脸的图像&#xff0c;是不是可以通过深度学习判别其是否是相…

聪明的人脸识别3——Pytorch 搭建自己的Facenet人脸识别平台

聪明的人脸识别3——Pytorch 搭建自己的Facenet人脸识别平台 学习前言什么是Facenet源码下载Facenet的实现思路一、预测部分1、主干网络介绍2、根据初步特征获得长度为128的特征向量3、l2标准化4、构建分类器&#xff08;用于辅助Triplet Loss的收敛&#xff09; 二、训练部分1…

syscall()

1、syscall的定义 #include<unistd.h> #include<sys/syscall.h> / For SYS_xxx definitions /long syscall(long number, ...);syscall执行间接系统调用&#xff0c;使用该函数会执行一个系统调用&#xff0c;根据指定的参数 number 和所有系统调用的汇编语言接口…

linux systemctl命令详解

笔者在前文中概要的介绍了 systemd 的基本概念和主要特点。由于 systemd 相关的绝大多数任务都是通过 systemctl 命令管理的&#xff0c;所以本文将集中的介绍 systemctl 命令的用法。注意&#xff0c;本文以 ubuntu 16.04 进行介绍&#xff0c;文中所有的 demo 都在 ubuntu 16…

Linux常用命令——sysctl命令

在线Linux命令查询工具(http://www.lzltool.com/LinuxCommand) sysctl 时动态地修改内核的运行参数 补充说明 sysctl命令被用于在内核运行时动态地修改内核的运行参数&#xff0c;可用的内核参数在目录/proc/sys中。它包含一些TCP/ip堆栈和虚拟内存系统的高级选项&#xff…

Linux之systemctl命令基本使用

文章目录 1. systemctl 管理指令2. systemctl 设置服务的自启动状态3. 应用案例&#xff1a;4. 细节讨论&#xff1a; 1. systemctl 管理指令 基本语法&#xff1a; systemctl [start | stop | restart | status] 服务名systemctl 指令管理的服务在 /usr/lib/systemd/system 查…

systemctl命令解析

原文链接如果有效&#xff0c;请点击原文链接查看。 原文&#xff1a;http://www.ruanyifeng.com/blog/2016/03/systemd-tutorial-commands.html 一、由来 历史上&#xff0c;Linux 的启动一直采用init进程。 下面的命令用来启动服务。 $ sudo /etc/init.d/apache2 start # …

Linux常用命令——systemctl命令

在线Linux命令查询工具(http://www.lzltool.com/LinuxCommand) systemctl 系统服务管理器指令 补充说明 systemctl命令是系统服务管理器指令&#xff0c;它实际上将 service 和 chkconfig 这两个命令组合到一起。 任务旧指令新指令使某服务自动启动chkconfig --level 3 ht…

【Linux】之systemd与systemctl

文章目录 一、systemd1. systemd 守护进程管理 Linux 的启动2. systemd 提供的功能:3. systemd 使用单元来管理不同类型的对象。4. 服务单元信息 二、systemctl1. systemctl输出中的服务状态2. 列出servera上安装的所以服务单元3. 列出servera上所有活动和不活动的套接字单元4.…

【linux systemctl】Linux命令之systemctl命令

一、systemctl命令简介 CentOS 5使用SysV init&#xff1b;CentOS 6使用Upstart&#xff0c;CentOS 7使用Systemd管理守护进程。centos7采用 systemd管理&#xff0c;服务独立的运行在内存中&#xff0c;服务响应速度快&#xff0c;但占用更多内存。独立服务的服务启动脚本都在…

Linux命令之systemctl命令

一、systemctl命令简介 CentOS 5使用SysV init&#xff1b;CentOS 6使用Upstart&#xff0c;CentOS 7使用Systemd管理守护进程。centos7采用 systemd管理&#xff0c;服务独立的运行在内存中&#xff0c;服务响应速度快&#xff0c;但占用更多内存。独立服务的服务启动脚本都在…

idea 如何使用tomcat启动项目

1、首先对项目进行打包&#xff08;使用maven&#xff09; 2、打包成功后会有target 3、选择右侧的长条框&#xff08;如果没有tomcat&#xff0c;框是灰色的&#xff0c;这时你要自己手动加一个tomcat&#xff09;&#xff0c;选择Edit 4、如图所示、点击“”号&#xff0c;继…

Tomcat启动项目慢

原因是多方面&#xff0c;我遇到的情况有三种可能导致tomcat启动项目变慢 情况一&#xff1a;tomcat在启动过程中会检查jar&#xff0c;当有大量的jar被检测的时候&#xff0c;启动需要很长时间 解决办法&#xff1a;将catalina.properties文件中的这一行 tomcat.util.scan.S…