当前位置: 首页 > news >正文

11 OpenCV图像识别之人脸识别

文章目录

  • 1 Eigenfaces
    • 1.1 建模流程
    • 1.2 示例代码
  • 2 Fisherfaces
    • 2.1 建模流程
    • 2.2 示例代码
  • 3 Local Binary Histogram
    • 3.1 建模流程
    • 3.2 示例代码

OpenCV 提供了三种人脸识别方法:

  1. Eigenfaces

Eigenfaces是一种基于PCA(Principal Component Analysis,主成分分析)的人脸识别方法,属于OpenCV中的特征脸方法之一。该方法将人脸图像转换为低维的特征向量,使用PCA降维的方式提取出训练集中的主成分特征,进而提取出人脸图像的特征向量。在进行识别时,通过比较输入图像与训练集中每个图像的特征向量的相似度来判断其所属的人脸类别。

  1. Local Binary Patterns Histograms (LBPH)

LBPH 是一种基于局部二值模式(Local Binary Pattern)的人脸识别方法,它将每个像素与周围的像素进行比较,计算出每个像素点的二进制编码,再将编码串联起来形成一个局部特征。LBPH 的优点是对于图像的旋转、缩放和灰度变化等不敏感,但对于遮挡和表情变化等因素的鲁棒性还有待提高。

  1. Fisherfaces

Fisherfaces 是一种基于线性判别分析(Linear Discriminant Analysis)的人脸识别方法,它通过将图像投影到低维空间,将图像从高维特征向量转换为低维特征向量,并使用低维特征向量进行分类。Fisherfaces 的优点是对于光照、表情变化等因素的鲁棒性很强,但对于遮挡的鲁棒性还有待提高。

1 Eigenfaces

Eigenfaces 是一种基于主成分分析(Principal Component Analysis, PCA)的人脸识别方法,它是最早的、也是最简单的人脸识别方法之一。

Eigenfaces 方法将每个人脸图像看作一个高维向量,通过对这些向量进行 PCA 变换,得到一组特征向量(也称为 eigenfaces),并将每个人脸图像投影到这组特征向量上,从而将高维的人脸数据降维到一个低维的特征空间中,进而实现人脸识别。

1.1 建模流程

首先,通过cv2.face.EigenFaceRecognizer_create()创建Eigenfaces识别器:

EigenFaceRecognizer = cv2.face.EigenFaceRecognizer_create([, num_components[, threshold]])

其中,可选参数num_components表示PCA保留的特征数量,默认为0,即保留所有特征。可选参数threshold表示一个阈值,当人脸图像与训练数据的差异小于该阈值时,认为这是同一个人。默认值为0,表示不使用阈值。

该方法返回一个EigenFaceRecognizer对象,可以使用其train()方法进行训练,predict()方法进行预测,save()load()方法保存和加载模型。

随后,进行训练:

cv2.face.EigenFaceRecognizer_create().train(src, labels)
或
EigenFaceRecognizer.train(src, labels)

其中,

  • src:训练数据,可以是一个列表或一个 NumPy 数组,其中每个元素都是包含人脸图像的数组。
  • labels:与训练数据对应的标签,传入的一个 NumPy 数组,其中每个元素都是一个整数标签,表示训练数据中对应人脸的身份信息。

该函数不返回任何值,训练后的识别器会保存在类实例中,可以通过调用 cv2.face.EigenFaceRecognizer_create().predict() 函数使用它。

最后,使用模型进行预测:

label, confidence = recognizer.predict(src)

recognizer.predict(src)函数是使用已经训练好的人脸识别器对输入的人脸图像进行识别。它接收一个参数src,即待识别的人脸图像。该函数返回两个值:

  • label:表示识别结果对应的人脸标签,通常是一个整数。
  • confidence:表示识别结果的置信度或准确度。值越小表示置信度越高,一般情况下,评分小于5000就可以认为匹配度高,评分为0则为完全一样。

1.2 示例代码

下面提供一个代码脚本,大概的流程为:

  1. 定义一个空列表 photos 和一个空列表 labels
  2. 使用 OpenCV 中的 cv2.imread() 函数读取一组人脸图像,并将其添加到 photos 列表中。每张图像都有一个对应的标签,表示这张图像属于哪个人。
  3. 创建一个字典 names,将每个标签与对应的人名关联起来。
  4. 使用 OpenCV 中的 cv2.face.EigenFaceRecognizer_create() 函数创建一个特征脸识别器。
  5. 使用 recognizer.train() 函数训练识别器,让其学会如何识别这组人脸图像。
  6. 读取一张待识别的人脸图像,并使用 recognizer.predict() 函数对其进行识别。该函数返回一个标签和一个置信度评分。
  7. 打印出评分和对应的人名。

训练数据网上可以较为容易的找到,也可以博客下面留言获取。

import cv2  
import numpy as np  photos = list()  
lables = list()  
photos.append(cv2.imread("第一张图象地址,后面依此类推", 0))  
lables.append(0)  # 第1张图像对应的标签  
photos.append(cv2.imread("", 0))  
lables.append(0)  # 第2张图像对应的标签  
photos.append(cv2.imread("", 0))  
lables.append(0)  # 第3张图像对应的标签  photos.append(cv2.imread("", 0))  
lables.append(1)  # 第4张图像对应的标签  
photos.append(cv2.imread("", 0))  
lables.append(1)  # 第5张图像对应的标签  
photos.append(cv2.imread("", 0))  
lables.append(1)  # 第6张图像对应的标签  names = {"0": "张三", "1": "李四"}  # 标签对应的名称字典  recognizer = cv2.face.EigenFaceRecognizer_create()  # 创建特征脸识别器  
recognizer.train(photos, np.array(lables))  # 识别器开始训练  i = cv2.imread("待识别.png", 0)  # 待识别的人脸图像  
label, confidence = recognizer.predict(i)  # 识别器开始分析人脸图像  
print("confidence = " + str(confidence))  # 打印评分  
print(names[str(label)])  # 数组字典里标签对应的名字  
cv2.waitKey()  
cv2.destroyAllWindows()

其运行结果为;
image.png

2 Fisherfaces

Fisherfaces 是一种基于线性判别分析(Linear Discriminant Analysis, LDA)的人脸识别方法,它是 Eigenfaces 方法的改进版本。

Fisherfaces 通过寻找最佳的投影方向(线性变换),将高维的人脸数据投影到一个低维的特征空间中,从而实现人脸识别。与 Eigenfaces 不同的是,Fisherfaces 不仅仅考虑样本之间的距离,还考虑了样本所属的类别之间的距离,因此更加适用于多类别人脸识别。

Fisherfaces 方法与 Eigenfaces 方法相比,具有更高的识别率和更好的鲁棒性,但计算成本较高。

2.1 建模流程

OpenCV的各模型建模流程相同,除

recognizer = cv2.face.EigenFaceRecognizer_create()
# 需要改成
recognizer = cv2.face.FisherFaceRecognizer_create()

其余均不需更改。

2.2 示例代码

示例代码与1.2中基本完全相同,需要更改的就是2.1中提到的部分。
更改方法后,运行结果为:
image.png
此处看出这个模型的confidence较高,也就是对结果更没有信心,其原因可能是因为 Fisherfaces 方法对样本数据的数量和质量的要求较高,以及对输入图像的光照、姿态、表情等变化更加敏感。

具体来说,Fisherfaces 方法需要在训练集中包含足够多的样本,并且每个样本需要具有代表性,才能够获得较好的分类效果。此外,Fisherfaces 方法对于输入图像的光照、姿态、表情等变化比较敏感,这些因素会影响特征向量的提取和分类结果的准确性,从而降低置信度。

相比之下,Eigenfaces 方法虽然对于样本数量和质量的要求不高,也比较容易实现,但其在人脸识别方面的分类能力和鲁棒性较差,因此在某些情况下可能会出现识别错误的情况,从而导致置信度较低。

3 Local Binary Histogram

3.1 建模流程

Local Binary Histogram(LBH)是一种基于局部二值模式(Local Binary Pattern, LBP)的人脸识别方法。与前面提到的基于 PCA 或 LDA 的人脸识别方法不同,LBH 方法直接利用局部纹理特征进行人脸识别,具有较好的鲁棒性和识别精度。

具体来说,LBH 方法通过以下步骤实现人脸识别:

  1. 对于每个人脸图像,将其分成若干个局部区域,并计算每个局部区域的 LBP 特征。

  2. 将每个局部区域的 LBP 特征用直方图统计的方法转换为一个向量。

  3. 将每个人脸图像的局部区域 LBP 特征向量组合成一个全局的特征向量,并将其归一化。

  4. 对于测试集中的每个人脸图像,同样将其分成若干个局部区域,并计算每个局部区域的 LBP 特征。将每个局部区域的 LBP 特征向量用直方图统计的方法转换为一个向量,并将这些向量组合成一个测试集的全局特征向量。

  5. 计算训练集中每个人脸图像的全局特征向量与测试集中人脸图像的全局特征向量之间的距离,将距离最小的训练集中的人脸图像所属的类别作为测试集中人脸图像的识别结果。

相比于传统的基于 PCA 或 LDA 的人脸识别方法,LBH 方法更加鲁棒,能够更好地应对光照、姿态、表情等变化,同时具有较高的识别率和较低的计算成本,因此被广泛应用于人脸识别和相关领域。

3.2 示例代码

此处需要更改的部分:

recognizer = cv2.face.LBPHFaceRecognizer_create()

此处因计算方法不同,需注意返回值confidence的含义与前两者不同。
此模型的返回值,小于50认为匹配度较高,0则为完全一样。

代码的输出为:
image.png

相关文章:

11 OpenCV图像识别之人脸识别

文章目录1 Eigenfaces1.1 建模流程1.2 示例代码2 Fisherfaces2.1 建模流程2.2 示例代码3 Local Binary Histogram3.1 建模流程3.2 示例代码OpenCV 提供了三种人脸识别方法:Eigenfaces Eigenfaces是一种基于PCA(Principal Component Analysis&#xff0c…...

ssh设置:免密登入、修改默认端口、禁止root登入、限制错误登入次数

服务器: 客户端: 在下面不再说明服务器和客户端。 1.修改ssh默认端口 是在服务器中设置。 该设置涉及三部分:sshd配置文件修改/增加新端口、Selinux添加新端口、Firewall开放新端口。 vim /etc/ssh/sshd.config,找到#Port行&…...

【Fastdfs】| 入门连续剧——安装

作者:狮子也疯狂 专栏:《spring开发》 坚持做好每一步,幸运之神自然会降临在你的身上 目录一. 🦁 前言Ⅰ. 🐇 为什么要使用分布式文件系统?1.1 单机系统 vs 独立文件服务器1.2 分布式文件系统1.3 FastDFS引…...

【ESP32-S3】Pycharm 使用 microPython 教程(避坑)

一、下载Pycharm等操作 1.百度云下载链接 链接:https://pan.baidu.com/s/1tkbMzS5B_v-Cn4WQlTqS3Q?pwd0108 提取码:0108 2.安装 按照压缩包中的教程来,你懂的。 二、配置microPython环境 1.安装 microPython 插件 1.1 File > Sett…...

Allegro如何通过报表的方式检查单板上是否有假器件操作指导

Allegro如何通过报表的方式检查单板上是否有假器件操作指导 在做PCB设计的时候,输出生产文件之前,必须保证PCB上不能存在假器件,如下图,是不被允许的 当PCB单板比较大,如何通过报表的方式检查是否存在假器件,具体操作如下 点击Tools点击Reports...

清理bib文件(删除重复项,仅保留tex中引用的条目)

在写latex文件的过程中,经常会遇到添加了一堆文献的bibtex到bib文件中,有时候文章一长同一篇文献用不同的cite-key引用了多次,同时也会有一些文献最后并没被正文引用,这就需要对bib文件进行清理。 删除重复项 可以用JabRef 在J…...

Rust编程细节知识点拾遗

1.Rust中每一个引用都有生命周期,也就是引用保持有效的作用域。生命周期主要目标是避免悬垂引用,悬垂引用就是引用了已经释放的值。函数中,x的生命周期不能小于返回值得生命周期。当有x和y的时候,两者的生命周期是两个里面较小的那…...

【Linux】线程池

🎇Linux: 博客主页:一起去看日落吗分享博主的在Linux中学习到的知识和遇到的问题博主的能力有限,出现错误希望大家不吝赐教分享给大家一句我很喜欢的话: 看似不起波澜的日复一日,一定会在某一天让你看见坚持…...

运动版蓝牙耳机什么牌子的好、运动款蓝牙耳机推荐

何以解忧?唯有运动。事实已经无数次证明,运动不但可以让你更瘦身、更紧实,更重要的是精神状态也能焕然一新。不知道各位是不是也跟我一样,喜欢在运动的时候听着音乐。但是听音乐就需要有好的续航,否则运动一半没电了&a…...

MySQL中自带的数据库表相关介绍

mysql的自带数据库表主要有以下几个: (1)information_schema (2)performance_schema (3)mysql (4)sys (5)可能存在空数据库test 一、informa…...

【微信小程序】--注册小程序账号(一)

💌 所属专栏:【微信小程序开发教程】 😀 作  者:我是夜阑的狗🐶 🚀 个人简介:一个正在努力学技术的CV工程师,专注基础和实战分享 ,欢迎咨询! &#…...

Java多线程 - 利用Callable或CompletableFuture实现多线程异步任务执行

文章目录1. Callable接口源码2. Future接口的源码3. RunnableFuture接口和FutureTask实现类4. 利用线程池和Callable接口实现异步执行任务5. 利用CompleteFutable实现多线程异步任务执行1. Callable接口源码 FunctionalInterface public interface Callable<V> {// 这个…...

【ts + webpack】贪吃蛇小游戏

目录 一、项目搭建 1.1 初始化项目 二、项目界面布局 三、完成Food类 四、完成记分牌类 五、初步完成snake类 六、创建游戏控制器类 - 键盘事件 七、GameControl - 使蛇移动 八、蛇撞墙和吃食检测 一、项目搭建 1.1 初始化项目 1.使用init命令生成package.json文件 …...

传统巨头生“变”,中国毫米波雷达市场战火再升级

进入2023年&#xff0c;中国车载毫米波雷达市场战火明显升级。 一方面&#xff0c;愈演愈烈的份额抢夺战不仅仅存在于几大传统巨头之间&#xff0c;也快速转移到与国产供应商之间&#xff1b;随着部分外资巨头的本土化战略深入落地&#xff0c;同时对国产供应商造成了压力。 …...

26岁曾月薪15K,现已失业3个月,我依然没有拿到offer......

我做测试5年&#xff0c;一线城市薪水拿到15K&#xff0c;中间还修了一个专升本&#xff0c;这个年限不说资深肯定也是配得上经验丰富的。今年行情不好人尽皆知&#xff0c;但我还是对我的薪水不是很满意&#xff0c;于是打算出去面试&#xff0c;希望可以搏一个高薪。 但真到面…...

华为OD机试 - 打印文件 | 机试题算法思路 【2023】

最近更新的博客 华为OD机试 - 简易压缩算法(Python) | 机试题算法思路 【2023】 华为OD机试题 - 获取最大软件版本号(JavaScript) 华为OD机试 - 猜字谜(Python) | 机试题+算法思路 【2023】 华为OD机试 - 删除指定目录(Python) | 机试题算法思路 【2023】 华为OD机试 …...

【前端】浏览器的渲染流程(完整)

本文主要包含以下内容&#xff1a;浏览器渲染整体流程解析 HTML样式计算布局分层生成绘制指令分块光栅化绘制常见面试题浏览器渲染整体流程浏览器&#xff0c;作为用户浏览网页最基本的一个入口&#xff0c;我们似乎认为在地址栏输入 URL 后网页自动就出来了。殊不知在用户输入…...

华为OD机试 - 有效子字符串 | 机试题算法思路 【2023】

最近更新的博客 华为OD机试 - 简易压缩算法(Python) | 机试题算法思路 【2023】 华为OD机试题 - 获取最大软件版本号(JavaScript) 华为OD机试 - 猜字谜(Python) | 机试题+算法思路 【2023】 华为OD机试 - 删除指定目录(Python) | 机试题算法思路 【2023】 华为OD机试 …...

抽象类和接口

抽象类和接口 抽象类和接口的定义 抽象类主要用来抽取子类的通用特性&#xff0c;作为子类的模板&#xff0c;它不能被实例化&#xff0c;只能被用作为子类的超类。 接口是抽象方法的集合&#xff0c;声明了一系列的方法操作&#xff0c;如果一个类实现了某个接口&#xff0c;…...

STM32DSP库汇总

前言 本文仅对stm32的DSP库进行汇总&#xff0c;具体函数使用方式持续更新…… 分类函数名描述 BasicMathFunctions 基础数学函数 abs绝对值add加法dot_prod向量点积mult乘法negate相反数offset 偏置 scale比例缩放shift移位sub减法 ComplexMathFunctions 复数数学函数 conj…...

后进先出(LIFO)详解

LIFO 是 Last In, First Out 的缩写&#xff0c;中文译为后进先出。这是一种数据结构的工作原则&#xff0c;类似于一摞盘子或一叠书本&#xff1a; 最后放进去的元素最先出来 -想象往筒状容器里放盘子&#xff1a; &#xff08;1&#xff09;你放进的最后一个盘子&#xff08…...

变量 varablie 声明- Rust 变量 let mut 声明与 C/C++ 变量声明对比分析

一、变量声明设计&#xff1a;let 与 mut 的哲学解析 Rust 采用 let 声明变量并通过 mut 显式标记可变性&#xff0c;这种设计体现了语言的核心哲学。以下是深度解析&#xff1a; 1.1 设计理念剖析 安全优先原则&#xff1a;默认不可变强制开发者明确声明意图 let x 5; …...

内存分配函数malloc kmalloc vmalloc

内存分配函数malloc kmalloc vmalloc malloc实现步骤: 1)请求大小调整:首先,malloc 需要调整用户请求的大小,以适应内部数据结构(例如,可能需要存储额外的元数据)。通常,这包括对齐调整,确保分配的内存地址满足特定硬件要求(如对齐到8字节或16字节边界)。 2)空闲…...

Spring Boot 实现流式响应(兼容 2.7.x)

在实际开发中&#xff0c;我们可能会遇到一些流式数据处理的场景&#xff0c;比如接收来自上游接口的 Server-Sent Events&#xff08;SSE&#xff09; 或 流式 JSON 内容&#xff0c;并将其原样中转给前端页面或客户端。这种情况下&#xff0c;传统的 RestTemplate 缓存机制会…...

基于服务器使用 apt 安装、配置 Nginx

&#x1f9fe; 一、查看可安装的 Nginx 版本 首先&#xff0c;你可以运行以下命令查看可用版本&#xff1a; apt-cache madison nginx-core输出示例&#xff1a; nginx-core | 1.18.0-6ubuntu14.6 | http://archive.ubuntu.com/ubuntu focal-updates/main amd64 Packages ng…...

【磁盘】每天掌握一个Linux命令 - iostat

目录 【磁盘】每天掌握一个Linux命令 - iostat工具概述安装方式核心功能基础用法进阶操作实战案例面试题场景生产场景 注意事项 【磁盘】每天掌握一个Linux命令 - iostat 工具概述 iostat&#xff08;I/O Statistics&#xff09;是Linux系统下用于监视系统输入输出设备和CPU使…...

镜像里切换为普通用户

如果你登录远程虚拟机默认就是 root 用户&#xff0c;但你不希望用 root 权限运行 ns-3&#xff08;这是对的&#xff0c;ns3 工具会拒绝 root&#xff09;&#xff0c;你可以按以下方法创建一个 非 root 用户账号 并切换到它运行 ns-3。 一次性解决方案&#xff1a;创建非 roo…...

C++中string流知识详解和示例

一、概览与类体系 C 提供三种基于内存字符串的流&#xff0c;定义在 <sstream> 中&#xff1a; std::istringstream&#xff1a;输入流&#xff0c;从已有字符串中读取并解析。std::ostringstream&#xff1a;输出流&#xff0c;向内部缓冲区写入内容&#xff0c;最终取…...

3403. 从盒子中找出字典序最大的字符串 I

3403. 从盒子中找出字典序最大的字符串 I 题目链接&#xff1a;3403. 从盒子中找出字典序最大的字符串 I 代码如下&#xff1a; class Solution { public:string answerString(string word, int numFriends) {if (numFriends 1) {return word;}string res;for (int i 0;i &…...

【JavaWeb】Docker项目部署

引言 之前学习了Linux操作系统的常见命令&#xff0c;在Linux上安装软件&#xff0c;以及如何在Linux上部署一个单体项目&#xff0c;大多数同学都会有相同的感受&#xff0c;那就是麻烦。 核心体现在三点&#xff1a; 命令太多了&#xff0c;记不住 软件安装包名字复杂&…...