身份证识别(一)——身份证正反面与头像检测

前言

1、这是一个手持的身份识别项目,属于图像识别的范围,不需要借助手持身份扫描器,只是检测当前视频或者摄像头中是否有身份证,然后做相关的处理。
2.这个项目是我之前做的一个大项目的一部分,项目用到的库有OpenCV,Boost, 还Caffe这个深度学习框架,用的一些相关的技术有图像处理的人脸检测,人脸识别,人脸验证,数字识别,汉字识别,以后有时间,可能会加上语音识别,或者对应声纹识别的相关操作。
3.我的项目环境是Ubuntu,Qt Creator 5.9.OpenCV3.3,使用Caffe训练VGG,VGG_Face,OCR等。

身份证检测

1.资源准备
(1)我写了个爬虫从网上下载了一大堆有身证的图像,这些身份都是公开的或者演示用的随机生成的身份证,所以应该不会某个人的个人隐私权,如果有侵犯,请私信我,下面是我下载的身份证的相关图像:
在这里插入图片描述
(2)使用LabelImage标注数据
在这里插入图片描述
(3)使用Caffe SSD的VGGNet训练模型,关于如何训练去调试参数,可以看我之前关于Caffe_SSD如何训练模型。
2.使用模型检测图像
(1)把以下三个文件导入到工程
在这里插入图片描述
(2)编写代码,进行识别。

void MainWindow::detectionId(Mat &input, string model_file,string model_text_file,string label_file)
{
    vector<String> objNames = readLabels(label_file);

    Ptr<dnn::Importer> importer;
    try
    {
        importer = createCaffeImporter(model_text_file, model_file);
    }
    catch (const cv::Exception &err)
    {
        cerr << err.msg << endl;
    }
    Net net;
    importer->populateNet(net);
    importer.release();

    Mat input_image = preprocess(input);
    Mat blobImage = blobFromImage(input_image);

    net.setInput(blobImage, "data");
    Mat detection = net.forward("detection_out");
    Mat detectionMat(detection.size[2], detection.size[3], CV_32F, detection.ptr<float>());
    float confidence_threshold = 0.2;
    for (int i = 0; i < detectionMat.rows; i++)
    {
        float confidence = detectionMat.at<float>(i, 2);
        if (confidence > confidence_threshold)
        {
            size_t objIndex = (size_t)(detectionMat.at<float>(i, 1));
            float tl_x = detectionMat.at<float>(i, 3) * input.cols;
            float tl_y = detectionMat.at<float>(i, 4) * input.rows;
            float br_x = detectionMat.at<float>(i, 5) * input.cols;
            float br_y = detectionMat.at<float>(i, 6) * input.rows;

            Rect object_box((int)tl_x, (int)tl_y, (int)(br_x - tl_x), (int)(br_y - tl_y));
            rectangle(input, object_box, Scalar(i*10, 0, 255), 2, 8, 0);
            putText(input, format("%s", objNames[objIndex].c_str()), Point(tl_x, tl_y), FONT_HERSHEY_SIMPLEX, 1.0, Scalar(255, 0, 0), 2);
        }
    }

}

Mat MainWindow::getMean(size_t &w, size_t &h)
{
    Mat mean;
    vector<Mat> channels;
    for (int i = 0; i < 3; i++)
    {
        Mat channel(h, w, CV_32F, Scalar(meanValues[i]));
        channels.push_back(channel);
    }
    merge(channels, mean);
    return mean;
}

Mat MainWindow::preprocess(Mat &frame)
{
    Mat preprocessed;
    frame.convertTo(preprocessed, CV_32F);
    cv::resize(preprocessed, preprocessed, Size(width, height));
    Mat mean = getMean(width, height);
    subtract(preprocessed, mean, preprocessed);
    return preprocessed;
}

3.识别结果
(1)打开图像

在这里插入图片描述
(2)检测当前图像中的身份证和身份证上的头像。
反面与头像
在这里插入图片描述
正面,这个图像多框了一个框。
在这里插入图片描述

结语

1.这是我用深度学习caffe_ssd的vgg做的身份证检测识别,也可以用传统的opencv方法去做。传统方法可以参考车牌识别项目。
2.关于整个工程的源码,运行程序时的bug,或者有如何优化的想法都可以加这个企鹅群:767133823 互相讨论学习
3.身份证边缘检测提取可以转到这个博客
4.yolov5训练可以转到这个博客
5.yolov5 C++ OpenCV DNN推理和最终模型以上传,想试试可以转这个博客

以下是一个简单的示例代码,演示如何自动裁剪身份证正反面图片: ```csharp using System; using System.Drawing; public static void AutoCropIDCard(string imagePath, string croppedImagePath) { // 加载原始图片 Image originalImage = Image.FromFile(imagePath); // 计算身份证的宽度和高度 int idCardWidth = 850; int idCardHeight = 540; // 计算正面和反面图片的左上角坐标和宽度和高度 int frontX = 30; int frontY = 74; int frontWidth = 388; int frontHeight = 276; int backX = 435; int backY = 74; int backWidth = 388; int backHeight = 276; // 创建一个新的身份证大小的图片 Image croppedImage = new Bitmap(idCardWidth, idCardHeight); // 创建一个 Graphics 对象,用于在新图片上绘制原始图片的一部分 Graphics graphics = Graphics.FromImage(croppedImage); // 定义一个矩形,表示要截取的正面图片区域 Rectangle frontSourceRectangle = new Rectangle(frontX, frontY, frontWidth, frontHeight); // 定义一个矩形,表示要绘制的正面图片目标区域 Rectangle frontDestinationRectangle = new Rectangle(0, 0, frontWidth, frontHeight); // 绘制原始图片的正面部分到新图片上 graphics.DrawImage(originalImage, frontDestinationRectangle, frontSourceRectangle, GraphicsUnit.Pixel); // 定义一个矩形,表示要截取的反面图片区域 Rectangle backSourceRectangle = new Rectangle(backX, backY, backWidth, backHeight); // 定义一个矩形,表示要绘制的反面图片目标区域 Rectangle backDestinationRectangle = new Rectangle(frontWidth, 0, backWidth, backHeight); // 绘制原始图片的反面部分到新图片上 graphics.DrawImage(originalImage, backDestinationRectangle, backSourceRectangle, GraphicsUnit.Pixel); // 保存新图片 croppedImage.Save(croppedImagePath, ImageFormat.Jpeg); // 释放资源 graphics.Dispose(); croppedImage.Dispose(); originalImage.Dispose(); } ``` 在这个示例中,`imagePath` 参数是原始图片的路径,`croppedImagePath` 参数是要保存的新图片的路径。 这个方法会读取原始图片,根据身份证正反面图片的位置和大小,自动裁剪出正反面图片,并将它们合并成一个新图片。最后,它会保存新图片到指定的路径。 您可以调用 `AutoCropIDCard` 方法来自动裁剪身份证图片。例如: ```csharp string imagePath = "path/to/image.jpg"; string croppedImagePath = "path/to/cropped/image.jpg"; AutoCropIDCard(imagePath, croppedImagePath); ``` 这个方法会读取原始图片,自动裁剪出身份证正反面图片,并将它们合并成一个新图片,并将结果保存到新图片中。
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

知来者逆

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值