OpenCV身份证离线识别技术的主要技术就是通过OpenCV找到身份证号码区域,然后通过OCR进行数字识别该区域的截图即可得到身份证号码。感兴趣的可以了解一下
4、膨胀与腐蚀
膨胀与腐蚀属于图像处理中最基本的形态学运算,形态学操作就是基于形状的一系列图像处理操作。OpenCV为进行图像的形态学变换提供了快捷且方便的函数。主要用于噪声消除、分割出独立的图像元素、在图像中连接相邻的元素、寻找图像中的明显的极大值区域或极小值区域、求出图像的梯度。
简单理解,膨胀就是求局部最大值的操作。腐蚀就是求局部最小值的操作。在处理身份证的时候,我们希望把身份证号码等数字区域连接在一起,即在图像中连接相邻的元素,所以需要使用膨胀处理,就跟蒸馒头的酵母粉一样,可以是我们想要的元素膨胀并且黏合在一起。
5、轮廓检测与图像分割
通过图像的膨胀操作,身份证号码区域已经被连接在一起了,目前需要做的事情就是检测出该区域的轮廓,使用拉普拉斯算子可以完成这个操作,OpenCV内部也提供了findContours函数做轮廓检测。
那么如何分割出身份证号码区域呢?其实有一个非常简单的思路,由于身份证号码是一串不换行的数字,宽高比通常是大于9:1的,而且是位于最后一行的,如果有其他的部分的宽高比大于9:1但是却不是位于最后,那么也不能认为是身份证号码,只有坐标是最底部,而且宽高比满足大于9:1的条件才可以。
主要代码
VS2022 + OpenCV4.5.4
#include <iostream>
#include <opencv2/opencv.hpp>
#include <vector>
#define DEFAULT_CARD_WIDTH 640
#define DEFAULT_CARD_HEIGHT 400
#define FIX_IDCARD_SIZE Size(DEFAULT_CARD_WIDTH, DEFAULT_CARD_HEIGHT)
#define FIX_TEMPLATE_SIZE Size(153, 28)
using namespace std;
using namespace cv;
int main() {
std::cout << "Hello, World!" << std::endl;
Mat src = imread("src.png");
imshow("src", src);
//处理身份证
Mat src_img = src;
//1、无损压缩 640*400 (通用卡片类的处理方式)
resize(src_img, src_img, FIX_IDCARD_SIZE);
imshow("dst", src_img);
Mat dst_img;
//2、灰度化
Mat dst;
cvtColor(src_img, dst, COLOR_BGR2GRAY);
imshow("gray", dst);
//3、二值化(降噪)
threshold(dst, dst, 100, 255, THRESH_BINARY);
imshow("threshold", dst);
// 4.1 腐蚀、膨胀
Mat erodeElement = getStructuringElement(MORPH_RECT, Size(20, 10));
erode(dst, dst, erodeElement);
imshow("erode", dst);
//4、轮廓检测,把所有的连续的闭包用矩形包起来
/*
* 一个矩形用两个点表示,contours就包含了很多矩形
*/
vector<vector<Point>> contours;
vector<Rect> rects;
findContours(dst, contours, RETR_TREE, CHAIN_APPROX_SIMPLE, Point(0, 0));
for (size_t i = 0; i < contours.size(); i++)
{
// 基于两点构建矩形
Rect rect = boundingRect(contours.at(i));
// 绘制矩形
rectangle(dst, rect, Scalar(0, 0, 255));
imshow("contours", dst);
// 对符合条件的图片进行筛选,宽高比大于1:9的
if(rect.width > rect.height*9)
{
cout << "找到了" << endl;
rects.push_back(rect);
rectangle(dst, rect, Scalar(0, 0, 255));
// 还需要再次矫正
//dst_img = src_img(rect);
}
}
// imshow("dst_Img", dst_img);
// 如果只找到了一个矩形,说明这个就是,如果多个就找出纵坐标最低的矩形
if(rects.size() == 1)
{
Rect rect = rects.at(0);
dst_img = src_img(rect);
}else
{
int lowPoint = 0;
Rect finalRect;
for (size_t i = 0; i < rects.size(); ++i)
{
Rect rect = rects.at(i);
Point p = rect.tl();
if(rect.tl().y > lowPoint)
{
lowPoint = rect.tl().y;
finalRect = rect;
}
}
rectangle(dst, finalRect, Scalar(255, 255, 0));
dst_img = src_img(finalRect);
}
imshow("dst_Img", dst_img);
waitKey();
return 0;
}
CMakeList.txt
cmake_minimum_required (VERSION 3.8)
project(opencv_idcard)
set(CMAKE_CXX_STANDARD 11)
add_executable (opencv_idcard "opencv_idcard.cpp" )
set(OpenCV_DIR "D:/develop/opencv-4.5.4/opencv-4.5.4-build")
find_package(OpenCV REQUIRED)
include_directories(${OpenCV_INCLUDE_DIRS})
target_link_libraries(opencv_idcard ${OpenCV_LIBS})
实现效果
来看看通过一系列的处理效果吧:
接下来要干的事情就主要有两件,首先是继承tess-two到Android,这样离线识别便搞定了,另外一件事情就是图像预处理的代码移植到Android上,这两件事情完成便搞定了身份证号码离线识别的功能了。
以上就是C++ OpenCV技术实战之身份证离线识别的详细内容,更多关于C++ OpenCV身份证离线识别的资料请关注编程学习网其它相关文章!
本文标题为:C++ OpenCV技术实战之身份证离线识别


- ubuntu下C/C++获取剩余内存 2023-09-18
- 详解C语言中sizeof如何在自定义函数中正常工作 2023-04-09
- C语言手把手带你掌握带头双向循环链表 2023-04-03
- C语言详解float类型在内存中的存储方式 2023-03-27
- C语言qsort()函数的使用方法详解 2023-04-26
- C++ 数据结构超详细讲解顺序表 2023-03-25
- 我应该为我的项目使用相对包含路径,还是将包含目录放在包含路径上? 2022-10-30
- c++ const 成员函数,返回一个 const 指针.但是返回的指针是什么类型的 const? 2022-10-11
- Easyx实现扫雷游戏 2023-02-06
- Qt计时器使用方法详解 2023-05-30