《深度学习》OpenCV 指纹验证、识别

目录

一、指纹验证

1、什么是指纹验证

2、步骤

1)图像采集

2)图像预处理

3)特征提取

4)特征匹配

5)相似度比较

6)结果输出

二、案例实现

1、完整代码

2、实现结果

调试模式:

三、指纹识别案例

1、展示指纹库图片

2、待验证指纹图

3、看完整代码

运行结果:


一、指纹验证

1、什么是指纹验证

        在OpenCV中,指纹验证是一种图像处理技术,用于识别和验证人类指纹。指纹是一种独特的生物特征,每个人的指纹都具有独特的纹路和图案。指纹验证使用这些独特的特征来确认一个人的身份。

        指纹验证主要包括两个步骤:指纹图像的提取指纹图像的匹配。

        在指纹图像提取阶段,OpenCV会处理输入的图像,通过一系列的图像处理特征提取算法,提取出指纹图像中的纹路和图案

        在指纹图像匹配阶段,OpenCV会将提取的指纹图像与一个或多个预先存储的指纹模板进行比对。比对过程中,OpenCV会计算两幅指纹图像之间的相似度,并根据相似度的阈值进行判断。

        如果两幅指纹图像的相似度超过了设定的阈值,OpenCV将判断它们属于同一个人,否则判断它们属于不同的人。指纹验证可以应用于许多领域,如安全系统、身份识别和刑事调查等。

2、步骤

        1)图像采集

                通过摄像头或扫描仪等设备获取人的手指指纹图像。

        2)图像预处理

                对采集到的指纹图像进行预处理,包括图像增强、去噪、增强对比度等操作,以便更好地提取指纹特征。

        3)特征提取

                在预处理后的图像中提取指纹的特征,常用的方法包括细化、方向计算、特征点定位等。

        4)特征匹配

                将提取的指纹特征与预先存储的指纹模板进行匹配。匹配算法可以使用比对指纹特征向量之间的相似度,如欧氏距离、汉明距离等。

        5)相似度比较

                根据匹配得到的相似度进行比较,判断两幅指纹图像是否属于同一个人。可以根据设定的阈值进行判断,超过阈值则认为匹配成功,否则认为匹配失败。

        6)结果输出

                根据匹配结果输出验证结果,可以是通过图像显示、文本信息或其他方式进行输出。

二、案例实现

1、完整代码

import cv2
def cv_show(name, img):
    cv2.imshow(name,img)
    cv2.waitKey(0)

def verification(src,model):   # 判断src图与模版图片model一致性
    # 创建SIFT特征提取器
    sift = cv2.SIFT_create()

    kp1,des1 = sift.detectAndCompute(src,None)  # 输入参数为图片、掩码图像,返回图片src的关键点坐标与关键点描述符
    kp2,des2 = sift.detectAndCompute(model, None)  # 计算模版图片的关键点和描述符

    # 创建FLANN匹配器,FLANN是一个高效的算法,用于在大规模数据集中执行最近邻搜索
    flann = cv2.FlannBasedMatcher()
    # 使用k近邻匹配(des1中的每个描述符与des2中的最近两个描述符进行匹配)
    # 对des1中的每个描述符在des2中查找两个最近邻
    matches = flann.knnMatch(des1,des2,k=2)   # 对待验证图与模版图进行匹配,返回匹配成功的点之间的欧式距离、测试图像的索引、样本图像的索引

# distance:匹配的特征点描述符的欧式距离,数值越小也就说明俩个特征点越相近。
# queryIdx:测试图像的特征点描述符的下标(第几个特征点描述符),同时也是描述符对应特征点的下标。
# trainIdx:样本图像的特征点描述符下标,同时也是描述符对应特征点的下标。
    # 进行比较筛选
    ok = []   # 存放匹配成功的点的坐标
    for m,n in matches:   # 遍历匹配成功点对应的两组欧氏距离,m为最近的一对点,n为次近的一对点
        # 根据lowe's比率测试,选择最佳匹配
        if m.distance < 0.8 * n.distance:  # 判断如果最近的比上次近的大小小于0.8,那么认为这是个正确的匹配
            ok.append(m)   # 将正确匹配的点存入列表
    # 统计逋过筛选的匹配数量
    num = len(ok)
    if num >= 500:   # 判断如果匹配数量大于500,则认为匹配成功
        result = "认证通过"
    else:
        result = "认证失败"
    return result


if __name__ == '__main__':
    src1 = cv2.imread("src1.BMP")   # 导入待验证图
    cv_show('src1', src1)
    src2 = cv2.imread("src2.BMP")
    cv_show('src2', src2)
    model = cv2.imread("model.BMP")   # 导入模版图
    cv_show('model',model)
    result1 = verification(src1,model)   # 放入函数进行判断
    result2 = verification(src2,model)
    print('src1验证结果为:',result1)
    print('src2验证结果为:',result2)

2、实现结果

调试模式:

三、指纹识别案例

1、展示指纹库图片

2、待验证指纹图

3、看完整代码

import re

import cv2
import numpy as np
import sys
import os


def getNum(src, model):  # 输入待验证图与模版图
    img1 = cv2.imread(src)
    img2 = cv2.imread(model)
    sift = cv2.SIFT_create()  # 创建sift特征提取器
    kp1,des1 = sift.detectAndCompute(img1, None)   # 提取待验证图片和模版图的关键点和描述符信息
    kp2,des2 = sift.detectAndCompute(img2, None)
    flann = cv2.FlannBasedMatcher()   # 建立Flann匹配器,其用来匹配大规模数据速度快
    matches = flann.knnMatch(des1,des2,k=2)   # 使用匹配器的K近邻算法匹配待匹配图片与模版图片,匹配两个最近距离
    ok = []
    for m,n in matches:   # 判断距离比例值是否小于0.8,是则将这一对点存入列表
        if m.distance < 0.8 *n.distance:
            ok.append(m)
    num = len(ok)    # 返回匹配成功的匹配数目
    return num


"""获取指纹编号"""

def getID(src, database):
    max = 0
    for file in os.listdir(database):  # 使用os.listdir读取database文件夹内的每一个文件
        model = os.path.join(database, file)   # 智能的将database的路径和file的路径结合成一个新的路径
        num = getNum(src,model)   # 将待验证图片src与提取出来的模版图model放入函数进行匹配,返回匹配成功的点的对数
        print("文件名:",file,"匹配数:",num)
        if num > max:  # 判断匹配成功的个数并不断更新max的值
            max = num   # 如果遇到最大匹配个数,那么将这个个数更新到max值,然后再更新模版图片的地址
            name = file
    ID = re.match(r'^(\d+)?\.([\S\s]+)$',name)[1]  # 正则匹配模版图片的文件名前缀
    if max < 100:   # src图片不一定是库里面人的指纹,判断匹配成功的数量是否小于100,小于则说明库里没有对应的指纹
        ID = 9999
    return ID   # 返回对应的图片名称

"""根据指纹编号,获取对应姓名"""
def getName(ID):
    nameID = {0:'张三',1:'李四',2:'王五',3:'赵六',4:'朱老七',5:'钱八',
                6:'曹九',7:'王二麻子',8:'andy',9:'Anna',9999:"没找到"}
    name = nameID.get(int(ID))
    return name

"""主函数"""
if __name__ == '__main__':
    src = "src.BMP"
    database = "database"
    ID = getID(src, database)
    name = getName(ID)   # 将得到的ID导入函数判断待验证指纹的人的姓名
    print("识别结果为:",name)
        运行结果:

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

SpringBoot(Java)实现MQTT连接(本地Mosquitto)通讯调试

1.工作及使用背景 工作中需要跟收集各种硬件或传感器数据用于Web展示及统计计算分析&#xff0c;如电表、流量计、泵、控制器等物联网设备。 目前的思路及解决策略是使用力控或者杰控等组态软件实现数据的转储&#xff08;也会涉及收费问题&#xff09;&#xff0c;通过组态软件…

Oracle数据恢复—异常断电导致Oracle数据库报错的数据恢复案例

Oracle数据库故障&#xff1a; 机房异常断电后&#xff0c;Oracle数据库启库报错&#xff1a;“system01.dbf需要更多的恢复来保持一致性&#xff0c;数据库无法打开”。数据库没有备份&#xff0c;归档日志不连续。用户方提供了Oracle数据库的在线文件&#xff0c;需要恢复zxf…

仕考网:事业单位考试选岗小技巧!

事业单位的考试选岗阶段&#xff0c;应该综合考量多个方面&#xff0c;确保选择出更合适的岗位&#xff0c;仕考网为大家分享以下技巧&#xff1a; 1. 岗位性质及工作内容 岗位性质:深入了解岗位是管理类、专业技术类还是工勤技能类&#xff0c;以及相应的职责和要求。 工作…

解决 Macos下 Orbstack docker网络问题

两种解决方法&#xff0c;第一种开代理 参考 —— 但是我这一种没成功&#xff0c;第二种方法是换镜像源 { "registry-mirrors": ["http://hub-mirror.c.163.com","https://docker.mirrors.ustc.edu.cn","https://mirrors.tencent.com&q…

openKylin--安装 .net6.0

编辑profile文件 cd .. //切换到根目录 cd /etc //切换到etc目录 vim profile //b编辑profile文件 1. 按→键移动到文件末尾 2. 按Insert键进入编辑模式 3. 按Enter另起一行开始编辑 export DOTNET_ROOT/home/dotnetexport PATH$PATH:/home/dotnet 可以通过右键--粘贴 的…

【Android 14源码分析】WMS-窗口显示-第二步:relayoutWindow -1

忽然有一天&#xff0c;我想要做一件事&#xff1a;去代码中去验证那些曾经被“灌输”的理论。                                                                                  – 服装…

YOLOv11改进策略【损失函数篇】| Slide Loss,解决简单样本和困难样本之间的不平衡问题

一、本文介绍 本文记录的是改进YOLOv11的损失函数&#xff0c;将其替换成Slide Loss&#xff0c;并详细说明了优化原因&#xff0c;注意事项等。Slide Loss函数可以有效地解决样本不平衡问题&#xff0c;为困难样本赋予更高的权重&#xff0c;使模型在训练过程中更加关注困难样…

Docker版MKVtoolnix的安装及中文显示

本文是应网友 kkkhi 要求折腾的&#xff0c;只研究了 MKVtoolnix 的安装及中文显示&#xff0c;未涉及到软件的使用&#xff1b; 什么是 MKVtoolnix &#xff1f; MKVToolnix 是一款功能强大的多媒体处理工具&#xff0c;用于在 Linux、其他 Unix 系统和 Windows 上创建、修改和…

SpringBoot--为什么Controller是串行的?怎样才能并行?

原文网址&#xff1a;SpringBoot--为什么Controller是串行的&#xff1f;怎样才能并行&#xff1f;-CSDN博客 简介 本文介绍SpringBoot为什么Controller是串行的&#xff1f;在什么场景下才能并行执行&#xff1f; 大家都知道&#xff0c;SpringBoot的Controller按理是并行执…

1.2.1 HuggingFists安装说明-Linux安装

Linux版安装说明 下载地址 【GitHub】https://github.com/Datayoo/HuggingFists 【百度网盘】https://pan.baidu.com/s/12-qzxARjzRjYFvF8ddUJQQ?pwd2024 安装说明 环境要求 操作系统&#xff1a;CentOS7 硬件环境&#xff1a;至少4核8G&#xff0c;系统使用Containerd…

IIS HTTPS 网页可能暂时无法连接,或者它已永久性地移动到了新网址 ERR_HTTP2_INADEQUATE_TRANSPORT_SECURITY

问题描述&#xff1a;站点突然无法访问&#xff0c;经排查发现&#xff0c;HTTP协议的网址可以继续访问&#xff0c;HTTPS的网址不可以访问。 问题分析&#xff1a;在Windows更新和滚动之后&#xff0c;由于 HTTP/2&#xff0c;当站点启动了 HTTP/2 连接&#xff0c;会出现一个…

通过PHP获取商品详情

在电子商务的浪潮中&#xff0c;数据的重要性不言而喻。商品详情信息对于电商运营者来说尤为宝贵。PHP&#xff0c;作为一种广泛应用的服务器端脚本语言&#xff0c;为我们提供了获取商品详情的便捷途径。 了解API接口文档 开放平台提供了详细的API接口文档。你需要熟悉商品详…

【JavaEE初阶】网络原理

欢迎关注个人主页&#xff1a;逸狼 创造不易&#xff0c;可以点点赞吗~ 如有错误&#xff0c;欢迎指出~ 目录 ⽹络互连 IP地址 端口号 协议 协议分层 优势 TCP/IP 五层网络模型 数据在网络通信中的整体流程 封装和分用 封装 分用 ⽹络互连 随着时代的发展&#xff0c;越来越需…

若依框架使用教程

1、若依介绍 1.1什么是低代码开发平台&#xff1f; 低代码诞生的目的是将可<font style"color:rgb(51, 51, 51);background-color:rgb(248, 248, 248);">重复性的编程</font>工作通过<font style"color:rgb(51, 51, 51);background-color:rgb(2…

.Net 6.0 监听Windows网络状态切换

上次发了一个文章获取windows网络状态&#xff0c;判断是否可以访问互联网。传送门&#xff1a;获取本机网络状态 这次我们监听网络状态切换&#xff0c;具体代码如下&#xff1a; public class WindowsNetworkHelper {private static Action<bool>? _NetworkStatusCh…

js列表数据时间排序和取唯一值

1.取唯一值[...new Set(array)] const array [1, 2, 3, 2, 4, 5, 3, 5]; // 使用Set去除重复元素 const uniarray [...new Set(array)]; console.log(uniarray); // 输出: [1, 2, 3, 4, 5] 2.排序 var u [1,3,2,5,4]; var uu u.sort(); console.log(uu); var u [1,3…

求组合数专题

求组合数 Ⅰ&#xff08;递推公式&#xff09; 思路 递推法预处理 利用公式 复杂度 直接查询 单次查询复杂度 代码 #include <bits/stdc.h> using namespace std; const int N 2010; const int mod 1e97; int c[N][N]; int get_c(int a, int b) {c[0][0] 1;for(i…

9700万个新岗位涌现,AI失业焦虑背后:超千万人找到了新工作

随着技术的飞速进步&#xff0c;全球就业市场正经历着翻天覆地的变化。 《2023年未来就业报告》预计&#xff0c;未来五年将新增近7000万个工作岗位&#xff0c;同时淘汰8300万个旧岗位。 在中国&#xff0c;城市如武汉和东莞正成为新就业机会的热土&#xff0c;无人驾驶、人…

C++ | Leetcode C++题解之第433题最小基因变化

题目&#xff1a; 题解&#xff1a; class Solution { public:int minMutation(string start, string end, vector<string>& bank) {int m start.size();int n bank.size();vector<vector<int>> adj(n);int endIndex -1;for (int i 0; i < n; i)…

数据结构2——单链表

在数据结构1——顺序表&#xff08;C语言版&#xff09;中&#xff0c;我们已经了解了顺序表的使用和实现&#xff0c;总结一下顺序表的优点&#xff1a; ①尾插尾删效率足够快&#xff1b; ②下标的随机访问和修改也足够方便。 可除此之外顺序表也确实存在着不足&#xff1a; …