OpenCV-Python 模板匹配的银行卡号识别

目录

  • 实现流程
  • 模板数字处理
    • 对模板图片进行二值化处理
    • 对二值化后的图片进行轮廓检测
    • 确定每个模板数字的图像
  • 银行卡图片处理
    • 预处理
    • 确定每组数字
    • 轮廓检测每个数字
    • 提取数字组中的数字
  • 模板匹配
  • 附录
    • 完整代码

实现流程

  1. 对数字模板进行处理,提取出单一数字的图片;

数字模板图片:
请添加图片描述
单一数字图片示例:
请添加图片描述

  1. 对银行卡图片进行预处理,提取出每组数字图像;

银行卡图片:

请添加图片描述

一组数字图片示例:
请添加图片描述

  1. 对每组数字图片,提取出单个数字图片,与模板图片进行模板匹配,寻找最优匹配结果,找到对应的数字。

单个数字图片示例:
请添加图片描述

模板数字处理

数字模板图片:
请添加图片描述

对模板图片进行二值化处理

#对模板二值化处理
ret , thre_temple = cv2.threshold(temple_gray, 200, 255, cv2.THRESH_BINARY_INV)

二值化后图片:
请添加图片描述

对二值化后的图片进行轮廓检测

#轮廓检测
temple_contours , hierarchy = cv2.findContours(thre_temple.copy(), cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_SIMPLE)
draw_con_temple = cv2.drawContours(temple, temple_contours, -1, (0, 0, 250), 1)
#print(np.array(temple_contours).shape)
cv_show('draw_con_temple', draw_con_temple)

轮廓检测效果:
请添加图片描述

确定每个模板数字的图像

根据各个轮廓寻找最小外接矩形,根据最小外接矩形的 y y y坐标的大小,对每个数字图片提取并存放在字典的对应序号中。

def get_contours_sort(contours):#找到外接矩形,根据y坐标大小boundingboxs = [cv2.boundingRect(cnt) for cnt in contours]bounding_boxs = sorted(boundingboxs)return bounding_boxs
#确定各个数字模板位置
bounding_boxs = get_contours_sort(temple_contours)#对每个模板提取并存储对应的数字
temple_difits = {}
for (i, c) in enumerate(bounding_boxs):(x, y, w, h) = croi = thre_temple[y:y+h , x:x+w ]#resize成合适大小roi = cv2.resize(roi, (50, 77))temple_difits[i] = roicv_show('temple_number', temple_difits[0])

单个数字图片示意:
请添加图片描述

银行卡图片处理

预处理

对银行卡图片进行各种预处理操作,滤除背景影响,同时使得每组数字连在一起;

#礼帽操作,突出更明亮的区域
tophat_goal = cv2.morphologyEx(goal, cv2.MORPH_TOPHAT, rectKernel)
cv_show('credit_card', tophat_goal)#sobel算子,边缘检测
gradX = cv2.Sobel(tophat_goal, ddepth = cv2.CV_32F, dx = 1, dy = 0)
gradX = np.abs(gradX)
(maxval , minval) = (np.max(gradX) , np.min(gradX))
gradX = (255*(gradX-minval)/(maxval-minval))
gradX = gradX.astype("uint8")
cv_show('sobel', gradX)#闭操作
close_goal = cv2.morphologyEx(gradX, cv2.MORPH_CLOSE, rectKernel)
cv_show('close', close_goal)#二值化
ret , thresh_goal = cv2.threshold(close_goal, 0, 255,  cv2.THRESH_OTSU)
cv_show('otsu', thresh_goal)#闭操作
close2_goal = cv2.morphologyEx(thresh_goal, cv2.MORPH_CLOSE, rectKernel)
cv_show('close2', close_goal)#二值化
ret , thre_goal = cv2.threshold(close2_goal, 200, 255, cv2.THRESH_BINARY)
cv_show('threshold', thre_goal)

要达到的效果示意:

请添加图片描述
如上图所示,各组数字连在了一起,同时背景也被滤掉。

确定每组数字

对预处理后的图片进行轮廓检测,根据轮廓外接矩形的长宽比确定每组数字位置。

#轮廓检测
goal_contours , hierarchy = cv2.findContours(thre_goal.copy(), cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_SIMPLE)
draw_con_goal = cv2.drawContours(color_goal.copy(), goal_contours, -1, (0, 0, 250), 1)
#print(np.array(temple_contours).shape)
cv_show('draw_con_goal', draw_con_goal)goal_boxs = find_goal_part(goal_contours)
num_of_num_part = goal_boxs.shape[0]
#print(goal_boxs)
#存储每个数字片段
goal_difits = {}
color_goal_difits = {}
for i in range(num_of_num_part):(x, y, w, h) = goal_boxs[i,:]roi = goal[y-5:y+h+5 , x-5:x+w+5 ]color_roi = color_goal[y-5:y+h+5 , x-5:x+w+5 ]color_goal_difits[i] = color_roigoal_difits[i] = roi
def find_goal_part(contours):#找到外接矩形,根据长宽比进行选取boundingboxs = [cv2.boundingRect(cnt) for cnt in contours]rate = []for (i , c) in enumerate(boundingboxs):(x, y, w, h) = crate.append(w/h)rate = np.array(rate)number_gath_id = np.argwhere((rate >= 3.55) & (rate <= 3.75))boundingboxs = np.array(boundingboxs)bounding_boxs =  boundingboxs[number_gath_id]bounding_boxs = bounding_boxs[:,0]sort_id = np.argsort(bounding_boxs[:,0])bounding_boxs = bounding_boxs[sort_id]return bounding_boxs

提取出的一组数字图片示例:
请添加图片描述

轮廓检测每个数字

对提取出的每一组图片进行二值化处理,并进行轮廓检测;

#对数字区域二值化处理
ret , num_part_temple = cv2.threshold(number_part, 200, 255, cv2.THRESH_BINARY)
cv_show('num_part_temple', num_part_temple)#轮廓检测
num_part_contours , hierarchy = cv2.findContours(num_part_temple.copy(), cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_SIMPLE)
draw_con_num_part = cv2.drawContours(color_num_part.copy(), num_part_contours, -1, (0, 0, 250), 1)
#print(np.array(draw_con_num_part).shape)
#cv_show('draw_con_num_part', draw_con_num_part)

二值化效果:

请添加图片描述

提取数字组中的数字

根据轮廓的外接矩形的 y y y坐标大小按顺序提取每个数字

#确定各个数字位置
num_part_boxs = get_contours_sort(num_part_contours)
num_in_num_part = np.array(num_part_boxs).shape[0]#对每个数字提取
num_part_difits = {}
for (i, c) in enumerate(num_part_boxs):(x, y, w, h) = croi = num_part_temple[y:y+h , x:x+w ]#resize成合适大小roi = cv2.resize(roi, (50, 77))cv_show('roi', roi)

提取出的数字:
请添加图片描述

模板匹配

对提取的数字进行模板匹配,最佳匹配图片序号即为当前图片数字。

for j in range(10):match_result[j] = cv2.matchTemplate(roi, temple_difits[j], cv2.TM_SQDIFF_NORMED)
min_val ,max_val ,min_id ,max_id = cv2.minMaxLoc(match_result)
result.append(min_id[1])

输出结果:
请添加图片描述

附录

完整代码

#识别银行卡上数字import cv2
import numpy as np
import matplotlib.pyplot as pltdef cv_show(name,img):cv2.imshow(name, img)cv2.waitKey(0)cv2.destroyAllWindows()def get_contours_sort(contours):#找到外接矩形,根据x位置排序boundingboxs = [cv2.boundingRect(cnt) for cnt in contours]bounding_boxs = sorted(boundingboxs)return bounding_boxsdef find_goal_part(contours):#找到外接矩形,根据长宽比进行选取boundingboxs = [cv2.boundingRect(cnt) for cnt in contours]rate = []for (i , c) in enumerate(boundingboxs):(x, y, w, h) = crate.append(w/h)rate = np.array(rate)number_gath_id = np.argwhere((rate >= 3.55) & (rate <= 3.75))boundingboxs = np.array(boundingboxs)bounding_boxs =  boundingboxs[number_gath_id]bounding_boxs = bounding_boxs[:,0]sort_id = np.argsort(bounding_boxs[:,0])bounding_boxs = bounding_boxs[sort_id]return bounding_boxstemple = cv2.imread('./orc_credit_num/temple.png' , 1)
temple_gray = cv2.imread('./orc_credit_num/temple.png' , 0)
goal = cv2.imread('orc_credit_num/credit_card_picture.png' , 0)
color_goal = cv2.imread('./orc_credit_num/temple.png' , 1)cv_show('temple', temple_gray)#对模板二值化处理
ret , thre_temple = cv2.threshold(temple_gray, 200, 255, cv2.THRESH_BINARY_INV)
cv_show('thre_temple', thre_temple)#轮廓检测
temple_contours , hierarchy = cv2.findContours(thre_temple.copy(), cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_SIMPLE)
draw_con_temple = cv2.drawContours(temple, temple_contours, -1, (0, 0, 250), 1)
#print(np.array(temple_contours).shape)
cv_show('draw_con_temple', draw_con_temple)#确定各个数字模板位置
bounding_boxs = get_contours_sort(temple_contours)#对每个模板提取并存储对应的数字
temple_difits = {}
for (i, c) in enumerate(bounding_boxs):(x, y, w, h) = croi = thre_temple[y:y+h , x:x+w ]#resize成合适大小roi = cv2.resize(roi, (50, 77))temple_difits[i] = roicv_show('temple_number', temple_difits[0])#对目标图像进行操作
#初始化卷积核
rectKernel = cv2.getStructuringElement(cv2.MORPH_RECT, (30,15))
sqKernel = cv2.getStructuringElement(cv2.MORPH_RECT, (5,5))cv_show('credit_card', goal)#礼帽操作,突出更明亮的区域
tophat_goal = cv2.morphologyEx(goal, cv2.MORPH_TOPHAT, rectKernel)
cv_show('credit_card', tophat_goal)#sobel算子,边缘检测
gradX = cv2.Sobel(tophat_goal, ddepth = cv2.CV_32F, dx = 1, dy = 0)
gradX = np.abs(gradX)
(maxval , minval) = (np.max(gradX) , np.min(gradX))
gradX = (255*(gradX-minval)/(maxval-minval))
gradX = gradX.astype("uint8")
cv_show('sobel', gradX)#闭操作
close_goal = cv2.morphologyEx(gradX, cv2.MORPH_CLOSE, rectKernel)
cv_show('close', close_goal)#二值化
ret , thresh_goal = cv2.threshold(close_goal, 0, 255,  cv2.THRESH_OTSU)
cv_show('otsu', thresh_goal)#闭操作
close2_goal = cv2.morphologyEx(thresh_goal, cv2.MORPH_CLOSE, rectKernel)
cv_show('close2', close_goal)#二值化
ret , thre_goal = cv2.threshold(close2_goal, 200, 255, cv2.THRESH_BINARY)
cv_show('threshold', thre_goal)#轮廓检测
goal_contours , hierarchy = cv2.findContours(thre_goal.copy(), cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_SIMPLE)
draw_con_goal = cv2.drawContours(color_goal.copy(), goal_contours, -1, (0, 0, 250), 1)
#print(np.array(temple_contours).shape)
cv_show('draw_con_goal', draw_con_goal)goal_boxs = find_goal_part(goal_contours)
num_of_num_part = goal_boxs.shape[0]
#print(goal_boxs)
#存储每个数字片段
goal_difits = {}
color_goal_difits = {}
for i in range(num_of_num_part):(x, y, w, h) = goal_boxs[i,:]roi = goal[y-5:y+h+5 , x-5:x+w+5 ]color_roi = color_goal[y-5:y+h+5 , x-5:x+w+5 ]color_goal_difits[i] = color_roigoal_difits[i] = roicv_show('goal_difits', goal_difits[3])
result = []
for k in range(num_of_num_part):#开始识别数字number_part = goal_difits[k]color_num_part = color_goal_difits[k]#对数字区域二值化处理ret , num_part_temple = cv2.threshold(number_part, 200, 255, cv2.THRESH_BINARY)cv_show('num_part_temple', num_part_temple)#轮廓检测num_part_contours , hierarchy = cv2.findContours(num_part_temple.copy(), cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_SIMPLE)draw_con_num_part = cv2.drawContours(color_num_part.copy(), num_part_contours, -1, (0, 0, 250), 1)#print(np.array(draw_con_num_part).shape)#cv_show('draw_con_num_part', draw_con_num_part)#确定各个数字位置num_part_boxs = get_contours_sort(num_part_contours)num_in_num_part = np.array(num_part_boxs).shape[0]#对每个数字提取num_part_difits = {}for (i, c) in enumerate(num_part_boxs):(x, y, w, h) = croi = num_part_temple[y:y+h , x:x+w ]#resize成合适大小roi = cv2.resize(roi, (50, 77))cv_show('roi', roi)match_result = np.zeros((10,1))for j in range(10):match_result[j] = cv2.matchTemplate(roi, temple_difits[j], cv2.TM_SQDIFF_NORMED)min_val ,max_val ,min_id ,max_id = cv2.minMaxLoc(match_result)result.append(min_id[1])print(result)    

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

银行卡验证接口API(仅需一行代码,实现全国银行卡核验)

银行卡确认在互联网越来越普及&#xff0c;在二手、电商、贷款、等主流互联网应用中都有明确的刚需&#xff0c;传统的 银行卡核验方式是用户上传图片&#xff0c;客服后台人肉审核&#xff0c;但问题在于根本无从确认 银行卡的真实性&#xff0c;也不能确认该 银行卡持有人和身…

openCV+Python实战练习——银行卡号识别

目录 项目Introduce&#xff1a; 项目名称&#xff1a; 具体操作步骤以及代码&#xff1a; 实现结果展示&#xff1a; 代码整体展示&#xff1a; 项目Introduce&#xff1a; 项目名称&#xff1a; 通过导入模板数字&#xff0c;对银行卡面上的数字进行识别&#xff0c;提…

【opencv学习】银行卡的识别

今天做一个小项目&#xff0c;运用已经学到的知识&#xff0c;做一个银行卡的识别。 银行卡的图像如下 这个银行卡是的数字模板是 现在我的任务是识别中间的16个数字 大致思路是&#xff1a; 1&#xff1a;读取模板&#xff0c;获得每个数字的模板信息 2&#xff1a;读取银行…

OpenCV C++案例实战二十《银行卡号识别》

OpenCV C案例实战二十《银行卡号识别》 前言一、获取模板图像1.1 功能效果1.2 功能源码 二、银行卡号定位2.1 将银行卡号切割成四块2.1.1 功能效果2.1.2 功能源码 2.2 字符切割2.2.1 功能效果2.2.2 功能源码 三、字符识别3.1.读取文件3.2.字符匹配3.3.功能源码 四、效果显示4.1…

opencv实际案例(一)银行卡号的识别

一、目标&#xff1a; 识别银行卡上的卡号 二、思路&#xff1a; 一 、准备模板&#xff0c;能够与卡号数字样式进行模型匹配 二、处理模板图片&#xff1a;将其分割成为单个数字的图片&#xff0c;和所对应的数字相匹配存储可迭代的数据类型中。 读入模板图片&#xff0…

(openCV--python)检测银行卡号 (应用篇)(包含模板图)

适合人群&#xff1a; 有python基础&#xff0c;注重应用、不追求深层次知识的小伙伴。前言&#xff1a; 本文对很多说用到的很多函数都是粗浅地解释一下用法以及作用&#xff0c;并不会解释该函数的具体结构。若想进一步深入了解还请自行谷狗一下。 完整代码 # Author :Eri…

opencv项目实践二(银行卡卡号识别)

任务说明 将银行卡卡号打印输出 实现 一、思路 1、定位银行卡卡号数字区域 2、将银行卡每个卡号数字单独提取出来 3、处理模板图片&#xff0c;将每个模板数字单独提取出来 4、通过模板匹配来识别每个卡号 二、具体代码实现 1、处理模板 卡号数字模板 # 读取模板图片 img_…

CVAT——2. CVAT简单使用

文章目录 基本使用管理员界面登录管理用户添加/删除用户及用户权限管理组管理 任务管理 创建任务和上传标签datumaroanaconda安装datumaro安装datumaro基本使用支持的格式查看帮助文件报错导入projects数据增加 数据标注标注面板使用快捷键左侧工具栏和右侧面板说明追踪模式Tra…

OpenCV图像处理——(实战)信用卡识别

总目录 图像处理总目录←点击这里 十四、信用卡数字识别 识别的图片模板图片 14.1、模板图片处理 读入图片->灰度图->二值图->计算轮廓->存储每一个模板 如果是所需模板匹配只有一个&#xff0c;课直接读入灰度图像即可 这里有10个模板(0-9)&#xff0c;所以需…

关于opencv的实战——银行卡号识别

首先我们先来引入我们所需要的库和我们定义的图像显示&#xff08;方便&#xff09;&#xff1a; import cv2 import numpy as np import imutils from imutils import contours#显示图像 def cv_show(img,name"image"):cv2.imshow(name,img)cv2.waitKey(0)cv2.dest…

13- 信用卡账号识别 (OpenCV基础) (项目十三) *

项目要点 _, ref cv2.threshold(ref, 10, 255, cv2.THRESH_BINARY_INV) 二值化处理图片, 黑白化图片ref cv2.cvtColor(img, cv2.COLOR_BGR2GRAY) 灰度化处理ref_contours, _ cv2.findContours(ref.copy(), cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_SIMPLE) 计算轮廓cv2.dr…

OpenCv图像处理实战——银行卡卡号识别

图像处理项目 测试图片自取区 card1 card2 银行卡卡号识别 from imutils import contours import numpy as np import argparse import imutils import cv2 import myutils #argparse这个库可以让我们直接在命令行中就可以向程序中传入参数并让程序运行。#这两个方法在包中没…

【OpenCV】 ⚠️实战⚠️ 银行卡卡号读取 ☢️建议手收藏☢️

【OpenCV】 ⚠️实战⚠️ 银行卡卡号读取 概述预处理代码模板预处理银行卡预处理 计算轮廓代码模板轮廓银行卡轮廓 其他程序主函数代码数字分割最终结果 概述 今天带大家使用我们之前学会的知识来实现银行卡卡号读取. 代码分为四个部分: 主函程序, 预处理, 计算轮廓, 其他程序…

openCV实践项目:银行卡卡号识别

本文用于对之前openCV知识点学习的复习及实践。要求达到以下效果&#xff1a; 一、基本流程思路分析 本项目本质上就是进行模板匹配。 注&#xff1a;为多用到所学知识&#xff0c;为了加深理解多加了些步骤&#xff0c;实际上本项目可以很简单就能完成。 1.1 模板处理 模…

什么是物联网平台

1、什么是物联网平台 阿里云物联网平台为设备提供安全可靠的连接通信能力&#xff0c;向下连接海量设备&#xff0c;支撑设备数据采集上云&#xff1b;向上提供云端API&#xff0c;服务端通过调用云端API将指令下发至设备端&#xff0c;实现远程控制。 物联网平台消息通信流程…

新兴 IoT 物联网场景中 MQTT 与 TCP 通信协议对比

在IoT 物联网开发中&#xff0c;大多数通信模组都支持 TCP、UDP、MQTT、CoAP、HTTP、LwM2M 等网络通信协议&#xff0c;其中既有传输层协议&#xff0c;也有应用层的协议&#xff0c;不同协议适用的场景也不同。 我们在设计IoT硬件产品时&#xff0c;通常只需选择一种协议即可。…

lot物联网场景通用架构分享

1.通用参考架构&#xff1a; 2.lot物联网硬件设备上云技术方案&#xff1a; #lot物联网业务链路包含&#xff1a;数据采集&#xff0c;通信连接&#xff0c;数据存储&#xff0c;数据可视化&#xff0c;洞察&#xff0c;行动决策&#xff1b;其中以设备端各种厂商提供得协议差…

物联网网络通讯知识

RTU是什么 RTU英文全称Remote Terminal Units&#xff0c;中文全称为远程终端单元。远程终端设备(RTU)是安装在远程现场的电子设备&#xff0c;用来监视和测量安装在远程现场的传感器和设备。通俗理解就是能够编程的还可以将数据传输到服务器的工具。RTU内部是包含通讯模块的&…

APP与服务器之间通过 http(POST、GET)进行数据交互 ( 实现一个简单的物联网系统-1 )

文章目录 一、APP POST 数据到服务器二、APP 从服务器 GET 数据三、APP 界面部分四、相关疑问五、学习方法六、专栏地址 一、APP POST 数据到服务器 首先 post 的数据应该包括识别这个花卉的信息和我们想要浇水的量&#xff0c;这里我们识别花卉采用花卉的名字&#xff0c;这样…

什么是物联网(Internet of Things)?

你可能在某些时候会听到物联网这个词&#xff0c;但是你对它可能不知所以然&#xff0c;这篇文章让你弄清楚什么是物联网&#xff0c;以及它代表什么&#xff1f; 物联网&#xff08;Internet of Things&#xff09;是指设备到互联网的连接。汽车&#xff0c;厨房电器&#xff…