OpenCV之信用卡识别

简介: 笔记

前言


在学完OpenCV中对图像的已经基本操作之后,例如彩色空间变换、阈值处理、图像梯度、轮廓检测、最小矩形匹配以及模板匹配。我们肯定非常急切地想去做一些事情,这里的信用卡卡号识别便是基于这些知识来做的!


正文


一.任务说明

在生活中,我们经常会遇到一些需要识别的地方,比如说在道路上的车牌识别、指纹识别、人脸识别等等,在不同的场景中所需要识别的内容也就不同。

在生活中的某一场景中(模拟),我们需要对银行卡卡号进行识别,来减轻我们工作的强度,需要我们设计算法,实现银行卡卡号的识别。


二.算法设计

1.数字的模板获取

这里我们的识别算法是根据模板匹配来实现的,在进行处理之前,需要准备与要识别信用卡数字风格差别不大的数字模板;模板的准备就不详细说明了,这也不是重点,模板数字图片如下。

对于准备的模板图片,还要将其中的每一个数字抠出来,保存在一个字典中,便于之后进行模板匹配。

def get_template(path):
    img = cv2.imread(path)    #读取数字模板
    img = cv2.resize(img,(400,64))      #resize到合适的大小
    gray = cv2.cvtColor(img,cv2.COLOR_BGR2GRAY)   #灰度处理
    _,gray = cv2.threshold(gray,100,255,cv2.THRESH_BINARY_INV)  #阈值处理
    cout,_ = cv2.findContours(gray,cv2.CV_8UC1,cv2.RETR_CCOMP)
#寻找轮廓
    point = []
    for i in cout:
        x, y, w, h = cv2.boundingRect(i)
        point.append((x,y,w,h))
        #将每一个数字对应的坐标保存起来
    point = sorted(point,key=lambda x:x[0],reverse=False)#对坐标进行排序,寻找轮廓时,顺序不是0-9的
    digit = {}
    for i,(x,y,w,h) in enumerate(point):
        digit[i] = cv2.resize(gray[y-2:y+h+2,x-2:x+w+2],(48,64))    #得到每个数字的图像
    return digit   #返回数字模板

2.信用卡卡号提取

  1. 对图片进行一些形态学操作
    1 构造适合的卷积核
    2 灰度处理
    3 礼帽操作
  #构造适合的卷积核
  rectKernel=cv2.getStructuringElement(cv2.MORPH_RECT,(9,3))
    sqKernel=cv2.getStructuringElement(cv2.MORPH_RECT,(5,5))
  gray = cv2.cvtColor(img,cv2.COLOR_BGR2GRAY)
    #礼帽操作
  gray = cv2.morphologyEx(gray,cv2.MORPH_TOPHAT,rectKernel)

3.求图形梯度

梯度操作,在这里非常重要,如果不进行梯度操作,信用卡上面的数字可能提取不全。

 #求x方向的梯度

 

gradX = cv2.Sobel(gray,ddepth=cv2.CV_32F,dx=1,dy=0,ksize=1)
    gradX = np.absolute(gradX)
    #归一化
    minVal,maxVal = np.min(gradX),np.max(gradX)
    gradX = (255*((gradX-minVal)/(maxVal-minVal)))
    gradX = gradX.astype("uint8")

4.重复一些形态学操作

1 闭操作

2 阈值处理

3 闭操作

gradX = cv2.morphologyEx(gradX,cv2.MORPH_CLOSE,rectKernel)
    _,thresh = cv2.threshold(gradX,127,255,cv2.THRESH_BINARY|cv2.THRESH_OTSU)
    thresh=cv2.morphologyEx(thresh,cv2.MORPH_CLOSE,sqKernel,iterations=2)


6.然寻找轮廓

1 将匹配到的轮廓安装适合的太小进行过滤,只保存我们需要的轮廓,数字

2 然后排序,按照原始顺序进行排列

group = []
    for i in cout:
        x,y,w,h = cv2.boundingRect(i)
        if(w/h)>3.2 and (w/h)<5:
            if (cv2.contourArea(i))>650 and (cv2.contourArea(i))<1000:
                group.append((x,y,w,h))
    #排序
    group = sorted(group,key=lambda x:x[0],reverse=False)

7.进行匹配

1 提取提取出来的信用卡数字,一一与模板进行匹配,以确定其数字是什么。

for x,y,w,h in group:
        img = gray[y-5:y+h+5,x-5:x+w+5]
        _, thresh = cv2.threshold(img, 127, 255, cv2.THRESH_BINARY | cv2.THRESH_OTSU)
        cout, _ = cv2.findContours(thresh, cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_SIMPLE)
        tt = []
        for i in cout:
            x,y,w,h = cv2.boundingRect(i)
            tt.append((x,y,w,h))
        tt = sorted(tt,key=lambda x:x[0],reverse=False)
        for x,y,w,h in tt:
            dst = img[y-2:y+h+2,x-2:x+w+2]
            dst = cv2.resize(dst,(48,64))
            sortes = []
            for j,temp in enumerate(digit.values()):
                result = cv2.matchTemplate(dst,temp,cv2.TM_CCOEFF)
                min_val, max_val, min_loc, max_loc = cv2.minMaxLoc(result)
                sortes.append(max_val)
            output.append(str(np.argmax(sortes)))


8.最后是画图操作

for i in range(4):
        put = ""
        for j in range(4):
            put +=output[i*4+j]
        cv2.putText(img,put,(dist[i][0],dist[i][1]-10),cv2.FONT_HERSHEY_SIMPLEX,0.8,(0,0,255),2)


三 总结

经过这个小项目的实践,让我更加熟悉了一些对图像处理的基本操作,也让我知道了这些基本操作组合起来的威力也是非常大的,也是可以处理实践问题的。

完整代码如下


import cv2
import numpy as np
"""
1. 模板
2. 二值化
3. 开闭操作
4. 轮廓匹配
5. 模板匹配
6. 顶帽
"""
#获取数字模板
def get_template(path):
    img = cv2.imread(path)
    img = cv2.resize(img,(400,64))
    gray = cv2.cvtColor(img,cv2.COLOR_BGR2GRAY)
    _,gray = cv2.threshold(gray,100,255,cv2.THRESH_BINARY_INV)
    cout,_ = cv2.findContours(gray,cv2.CV_8UC1,cv2.RETR_CCOMP)
    point = []
    for i in cout:
        x, y, w, h = cv2.boundingRect(i)
        point.append((x,y,w,h))
    point = sorted(point,key=lambda x:x[0],reverse=False)
    digit = {}
    for i,(x,y,w,h) in enumerate(point):
        digit[i] = cv2.resize(gray[y-2:y+h+2,x-2:x+w+2],(48,64))
    return digit
def get_digit_area(img,digit):
    #构造卷积核
    rectKernel=cv2.getStructuringElement(cv2.MORPH_RECT,(9,3))
    sqKernel=cv2.getStructuringElement(cv2.MORPH_RECT,(5,5))
    gray = cv2.cvtColor(img,cv2.COLOR_BGR2GRAY)
    #礼帽操作
    gray = cv2.morphologyEx(gray,cv2.MORPH_TOPHAT,rectKernel)
    cv2.imshow('a',gray)
    #求x方向的梯度
    gradX = cv2.Sobel(gray,ddepth=cv2.CV_32F,dx=1,dy=0,ksize=1)
    gradX = np.absolute(gradX)
    #归一化
    minVal,maxVal = np.min(gradX),np.max(gradX)
    gradX = (255*((gradX-minVal)/(maxVal-minVal)))
    gradX = gradX.astype("uint8")
    #闭操作
    cv2.imshow("gra",gradX)
    cv2.waitKey(0)
    gradX = cv2.morphologyEx(gradX,cv2.MORPH_CLOSE,rectKernel)
    _,thresh = cv2.threshold(gradX,127,255,cv2.THRESH_BINARY|cv2.THRESH_OTSU)
    thresh=cv2.morphologyEx(thresh,cv2.MORPH_CLOSE,sqKernel,iterations=2)
    #寻找轮廓
    cout,_ = cv2.findContours(thresh,cv2.RETR_EXTERNAL,cv2.CHAIN_APPROX_SIMPLE)
    group = []
    for i in cout:
        x,y,w,h = cv2.boundingRect(i)
        if(w/h)>3.2 and (w/h)<5:
            if (cv2.contourArea(i))>650 and (cv2.contourArea(i))<1000:
                group.append((x,y,w,h))
    #排序
    group = sorted(group,key=lambda x:x[0],reverse=False)
    output = []
    for x,y,w,h in group:
        img = gray[y-5:y+h+5,x-5:x+w+5]
        _, thresh = cv2.threshold(img, 127, 255, cv2.THRESH_BINARY | cv2.THRESH_OTSU)
        cout, _ = cv2.findContours(thresh, cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_SIMPLE)
        tt = []
        for i in cout:
            x,y,w,h = cv2.boundingRect(i)
            tt.append((x,y,w,h))
        tt = sorted(tt,key=lambda x:x[0],reverse=False)
        for x,y,w,h in tt:
            dst = img[y-2:y+h+2,x-2:x+w+2]
            dst = cv2.resize(dst,(48,64))
            sortes = []
            for j,temp in enumerate(digit.values()):
                result = cv2.matchTemplate(dst,temp,cv2.TM_CCOEFF)
                min_val, max_val, min_loc, max_loc = cv2.minMaxLoc(result)
                sortes.append(max_val)
            output.append(str(np.argmax(sortes)))
    return group,output
def main():
    img = cv2.imread("credit_card_02.png")
    img = cv2.resize(img,(400,250))
    digit = get_template("template.png")
    dist,output = get_digit_area(img,digit)
    for i in range(4):
        put = ""
        for j in range(4):
            put +=output[i*4+j]
        cv2.putText(img,put,(dist[i][0],dist[i][1]-10),cv2.FONT_HERSHEY_SIMPLEX,0.8,(0,0,255),2)
    print(output)
    cv2.imshow("A",img)
    cv2.waitKey(0)
if __name__ == '__main__':
    main()

公众号:FPGA之旅

目录
相关文章
|
C++ Python
Python+OpenCV人行道盲道边缘侦测识别
Python+OpenCV人行道盲道边缘侦测识别
224 1
Python+OpenCV人行道盲道边缘侦测识别
|
监控 计算机视觉 Python
用Python和OpenCV库实现识别人物出现并锁定
用Python和OpenCV库实现识别人物出现并锁定
283 0
|
计算机视觉 Python
最快速度写出一个识别效果——OpenCV模板匹配(含代码)
最快速度写出一个识别效果——OpenCV模板匹配(含代码)
544 0
|
8月前
|
算法 计算机视觉 开发者
OpenCV中使用Eigenfaces人脸识别器识别人脸实战(附Python源码)
OpenCV中使用Eigenfaces人脸识别器识别人脸实战(附Python源码)
399 0
|
计算机视觉 Python
OpenCV完成面部情绪识别
OpenCV完成面部情绪识别
235 0
|
6月前
|
计算机视觉 Python
opencv识别颜色
opencv识别颜色
|
6月前
|
机器学习/深度学习 数据采集 算法
Python基于OpenCV和卷积神经网络CNN进行车牌号码识别项目实战
Python基于OpenCV和卷积神经网络CNN进行车牌号码识别项目实战
|
8月前
|
计算机视觉 开发者 Python
OpenCV中Fisherfaces人脸识别器识别人脸实战(附Python源码)
OpenCV中Fisherfaces人脸识别器识别人脸实战(附Python源码)
295 0
|
7月前
|
移动开发 算法 计算机视觉
技术笔记:openCV特征点识别与findHomography算法过滤
技术笔记:openCV特征点识别与findHomography算法过滤
120 0
|
8月前
|
算法 计算机视觉
基于opencv的指针式仪表的识别与读数
基于opencv的指针式仪表的识别与读数