居民身份证阅读器产品开发学习心得(再谈标准-软件-协议)

本文涉及的产品
全局流量管理 GTM,标准版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
云解析 DNS,旗舰版 1个月
简介: 居民身份证阅读器产品开发学习心得(再谈标准-软件-协议)

说到一些产品化的东西,之前就写过一篇关于标准化的文章,当然作为我本人而言,也是在不断的学习中,理解标准,则有利于未来转型走向产品以及市场相关的岗位,因为我不单单是要了解怎么做?(这是技术层面),我还需要了解它为什么这么做?(业务层,产品化,商业化,价值),那能不能随便DIY就说这是一个产品?


谈谈做产品、做项目以及标准化相关的话题


今天来分享分享我主业安防军工相关的一些产品模块的使用心得,当然这不是涉密的东西,因为百度都可以搜得到,但是百度的东西缺少归类,所以写这篇文章的意义在于总结和分享,也同时希望从事安防、军工行业的大佬能和我一起多多交流心得。


回到正题,身份证阅读器产品必须符合中华人民共和国公共安全行业标准,既然是符合国标,那么就一定有软硬件接口技术规范,我们找来了编号GA467-2012这个文档,市面上很多身份证制具产品很多是基于该标准进行开发。


640.png

这份规范对身份证制具的研发和基于身份证制具进行开发的产品具有重要指导意义,它涵盖了软硬件接口的规范,也就是说,如果我们要开发这个设备,就必须按照这个规范去做,否则没有任何意义。市面上的身份证阅读制具一般有两种接口,分别是USBUART(包括5V CMOS电平RS-232C电平两种信号接口)。

1、通信接口说明

1.1、UART接口参数说明

640.png

1.2、USB接口参数说明

640.png

2、传输数据格式与协议分析

我们以市面上最常见的UART传输格式,它的基本传输格式如下:

640.png

以通俗易懂的方式去分析,这个传输格式的意思是,当我们发送指令给模块时,采用的是表10这种传输帧格式,当接 收模块给我们回复的数据时候采用的是表11这种数据输出传输帧格式。


2.1、以上传输格式各个字段的含义

640.png

2.2 命令集及应答码

640.png

这个命令集指的就是我们给身份证制具设备发送的指令。

640.png

这个SAM_A应答码指的是当用户发送指令给设备的时候,设备给我们返回的内容,根据内容我们就可以判断身份证 制具的状态,以及程序控制逻辑。

3、基于IDM20身份证阅读制具读取样例

640.png

一般安防产品类开发只会用到以下几个常见的命令,分别是复位SAM_A、SAM_A状态检测、寻找居民身份证、选取 居民身份证、读机读文字信息和相片信息,其它一般不会用到,除非是特殊的应用场景。

3.1 基于C#上位机Demo

之前在C#上实现了一个简单的读取身份证信息的上位机Demo,对以上各个指令做了封装,设计了如下简单的软件界 面,具体代码可百度自行参考。


各个指令的封装:

//复位SAM_A
private void Reset_SAM_A_Click(object sender, EventArgs e)
{
    byte[] cmd_format = new byte[10];
    cmd_format[0] = 0xAA;
    cmd_format[1] = 0xAA;
    cmd_format[2] = 0xAA;
    cmd_format[3] = 0x96;
    cmd_format[4] = 0x69;
    cmd_format[5] = 0x00;
    cmd_format[6] = 0x03;
    cmd_format[7] = 0x10;
    cmd_format[8] = 0xFF;
    cmd_format[9] = (byte)((byte)cmd_format[5] ^ (byte)cmd_format[6] ^ (byte)cmd_format[7] ^ (byte)cmd_format[8]);
    this.serialPort1.Write(cmd_format, 0, cmd_format.Length);
}
//SAM_A状态检测
private void SAM_A_STATUS_Click(object sender, EventArgs e)
{
    byte[] cmd_format = new byte[10];
    cmd_format[0] = 0xAA;
    cmd_format[1] = 0xAA;
    cmd_format[2] = 0xAA;
    cmd_format[3] = 0x96;
    cmd_format[4] = 0x69;
    cmd_format[5] = 0x00;
    cmd_format[6] = 0x03;
    cmd_format[7] = 0x11;
    cmd_format[8] = 0xFF;
    cmd_format[9] = (byte)((byte)cmd_format[5] ^ (byte)cmd_format[6] ^ (byte)cmd_format[7] ^ (byte)cmd_format[8]);
    this.serialPort1.Write(cmd_format, 0, cmd_format.Length);
}
//寻找居民身份证
private void Find_ID_Card_Click(object sender, EventArgs e)
{
    byte[] cmd_format = new byte[10];
    cmd_format[0] = 0xAA;
    cmd_format[1] = 0xAA;
    cmd_format[2] = 0xAA;
    cmd_format[3] = 0x96;
    cmd_format[4] = 0x69;
    cmd_format[5] = 0x00;
    cmd_format[6] = 0x03;
    cmd_format[7] = 0x20;
    cmd_format[8] = 0x01;
    cmd_format[9] = (byte)((byte)cmd_format[5] ^ (byte)cmd_format[6] ^ (byte)cmd_format[7] ^ (byte)cmd_format[8]);
    this.serialPort1.Write(cmd_format, 0, cmd_format.Length);
}
//选取居民身份证
private void Select_ID_Card_Click(object sender, EventArgs e)
{
    byte[] cmd_format = new byte[10];
    cmd_format[0] = 0xAA;
    cmd_format[1] = 0xAA;
    cmd_format[2] = 0xAA;
    cmd_format[3] = 0x96;
    cmd_format[4] = 0x69;
    cmd_format[5] = 0x00;
    cmd_format[6] = 0x03;
    cmd_format[7] = 0x20;
    cmd_format[8] = 0x02;
    cmd_format[9] = (byte)((byte)cmd_format[5] ^ (byte)cmd_format[6] ^ (byte)cmd_format[7] ^ (byte)cmd_format[8]);
    this.serialPort1.Write(cmd_format, 0, cmd_format.Length);
}
//读身份证信息
private void Read_ID_Card_Info_Click(object sender, EventArgs e)
{
    byte[] cmd_format = new byte[10];
    cmd_format[0] = 0xAA;
    cmd_format[1] = 0xAA;
    cmd_format[2] = 0xAA;
    cmd_format[3] = 0x96;
    cmd_format[4] = 0x69;
    cmd_format[5] = 0x00;
    cmd_format[6] = 0x03;
    cmd_format[7] = 0x30;
    cmd_format[8] = 0x01;
    cmd_format[9] = (byte)     ((byte)cmd_format[5] ^ (byte)cmd_format[6] ^ (byte)cmd_format[7] ^ (byte)cmd_format[8]);
    this.serialPort1.Write(cmd_format, 0, cmd_format.Length);
    Read_ID_Card_Info_Flag = true;
}

操作顺序:发送复位SAM_A指令===>将身份证放置在制具阅读区===>发送寻找居民身份证指令===>发送选取居民身 份证指令===>读身份证信息。


关于身份证信息解析,那么肯定也是有固定格式的,我们来看下:(以二代居民身份证为例)

640.png

按操作顺序,当我们发送读身份证信息指令时,设备会返回以上数据格式,我们只需要根据以上格式对各个字段进行解析即可,数据存储格式默认以Unicode的格式进行存放,所以我们需要以Unicode的存储格式对读取的数据进行解析。


在C# demo的串口回调上实现如下:

//设置串口接收回调
public void sp_DataRecevied(object sender, SerialDataReceivedEventArgs eg)
{
    System.Threading.Thread.Sleep(500);
    this.Invoke((EventHandler)delegate//异步执行 一个线程
    {
        StringBuilder sb = new StringBuilder();
        long rec_count = 0;
        int num = this.serialPort1.BytesToRead;
        byte[] recbuf = new byte[num];
        rec_count += num;
        this.serialPort1.Read(recbuf, 0, num);
        string str = " ";
        for (int i = 0; i < recbuf.Length; i++)
        {
            str += "0x" + Convert.ToString(recbuf[i], 16) + " ";
        }
        this.textBox1.AppendText(str);
        //如果是读取身份信息信息,则将所有的数据重定向到特定的缓冲区里进行处理
        if (Read_ID_Card_Info_Flag)
        {
            Read_ID_Card_Info_Flag = false;
            //文字信息长度
            int Text_Info_Length;
            //图像信息长度
            int Pic_Info_Length;
            Text_Info_Length = recbuf[10] << 8 | recbuf[11];
            Pic_Info_Length = recbuf[12] << 8 | recbuf[13];
            //身份证信息读取
            byte[] font_info = new byte[Text_Info_Length];
            for (int i = 0; i < Text_Info_Length; i++)
            {
                font_info[i] = recbuf[14 + i];
            }
            //获取姓名
            string __font_name = Encoding.Unicode.GetString(font_info, 0, 30).Trim();
            this.Name.Text = __font_name;
            //获得性别
            string __font_sex = Encoding.Unicode.GetString(font_info, 30, 2).Trim();
            //1表示性别男
            if (__font_sex.Contains("1"))
                this.Sex.Text = "男";
            //获得民族
            string __font_nation = Encoding.Unicode.GetString(font_info, 32, 4).Trim();
            //代号01是表示汉族,其它可自行查询
            if(__font_nation.Contains("01"))
                this.Nation.Text = "汉族";
            //获得生日
            string __font_birthday = Encoding.Unicode.GetString(font_info, 36, 16).Trim();
            this.Birthday.Text = __font_birthday;
            //获得住址
            string __font_address = Encoding.Unicode.GetString(font_info, 52, 70).Trim();
            this.Address.Text = __font_address;
            //获取身份证号码
            string __font_id_number = Encoding.Unicode.GetString(font_info, 122, 36).Trim();
            this.IDCard_Number.Text = __font_id_number;
            //获得签发机关
            string __font_Issuing_authority = Encoding.Unicode.GetString(font_info, 158, 30).Trim();
            this.Issuing_authority.Text = __font_Issuing_authority;
            //身份证照片读取
            byte[] img_info = new byte[Pic_Info_Length];
            for (int i = 0; i < img_info.Length; i++)
            {
                img_info[i] = recbuf[14 + Text_Info_Length + i];
            }
        }
        sb.Clear();
    });
}

最终效果如下:

640.png

3.2 基于STM32 Demo(小熊派验证)

640.png

关于指令集,我们可以用一个结构体进行封装:

/*数据包头 0xAA 0xAA 0xAA 0x96 0x69*/
#define CMD_HEADER_0 0xAA
#define CMD_HEADER_1 0xAA
#define CMD_HEADER_2 0xAA
#define CMD_HEADER_3 0x96
#define CMD_HEADER_4 0x69
/*命令和参数*/
#define CMD_RESET_SAM_A        0x01
#define CMD_RESET_SAM_A_PARA     0xFF
#define CMD_READ_SAM_A_STATUS     0x11
#define CMD_READ_SAM_A_STATUS_PARA 0xFF
#define CMD_FIND_ID_CARD       0x20
#define CMD_FIND_ID_CARD_PARA    0x01
#define CMD_SELECT_ID_CARD        0x20
#define CMD_SELECT_ID_CARD_PARA    0x02
#define CMD_READ_INFO        0x30
#define CMD_READ_INFO_PARA     0x01
/*业务终端通过业务终端接口发送的命令集数据结构*/
typedef struct
{
 /*命令*/
 uint8_t CMD ;  
 /*命令参数*/
 uint8_t CMD_PARA ;
}BUSSINESS_LIST ;

然后定义一个表:

/*IDM20身份证阅读机具命令表*/
BUSSINESS_LIST CMD_TABLE[] =
{
    /*复位SAM_A*/
    {CMD_RESET_SAM_A, CMD_RESET_SAM_A_PARA},
    /*SAM_A状态检测*/
    {CMD_READ_SAM_A_STATUS, CMD_READ_SAM_A_STATUS_PARA},
    /*寻找居民身份证*/
    {CMD_FIND_ID_CARD, CMD_FIND_ID_CARD_PARA},
    /*选取居民身份证*/
    {CMD_SELECT_ID_CARD, CMD_SELECT_ID_CARD_PARA},
    /*读机读文字信息和相片信息*/
    {CMD_READ_INFO, CMD_READ_INFO_PARA},
};

发送指令的时,要加上帧头以及其它部分,具体请参考数据传输格式:

/*命令包发送处理*/
static int CMD_Packet_Send_Handler(uint8_t CMD_NUMBER)
{
    uint8_t count = 0 ;
    uint8_t CMD_MERGE[CMD_LEN] = {0};
    uint8_t CMD_HEAD[CMD_HEAD_LEN] =
    {
        CMD_HEADER_0, CMD_HEADER_1,
        CMD_HEADER_2, CMD_HEADER_3,
        CMD_HEADER_4
    };
    if(CMD_NUMBER > 4)
      return -1 ;
    for(count = 0 ; count < CMD_HEAD_LEN ; count++)
        CMD_MERGE[count] = CMD_HEAD[count];
    CMD_MERGE[5] = 0x00 ;
    CMD_MERGE[6] = 0x03 ;
    CMD_MERGE[7] = CMD_TABLE[CMD_NUMBER].CMD;
    CMD_MERGE[8] = CMD_TABLE[CMD_NUMBER].CMD_PARA;
    CMD_MERGE[9] = CMD_MERGE[5] ^ CMD_MERGE[6] ^ CMD_MERGE[7] ^ CMD_MERGE[8];
    return HAL_UART_Transmit(&huart2, CMD_MERGE, 10, 0xff);
}

在串口回调接收中,采用DMA+空闲中断的方式接收回复。


注意: 身份证阅读制具属于被动设备,因为它不会主动上报,所以我们需要定时不断去发送寻卡指令,通过设备返回的状态字来确定制具上是否有身份证卡,如果有再进行选卡和读信息的操作。


如上,STM32的解析方法与C#解析方法类似,最终效果:

640.png

往期精彩

C语言常用的一些转换工具函数收集


结构体对齐原则在自定义协议解析时的妙用之法


【为宏正名】99%人都不知道的"##"里用法


【为宏正名】本应写入教科书的“世界设定”

目录
相关文章
|
10天前
|
物联网 测试技术 开发工具
开发 Bluegiga APX4 协议产品需要哪些技术知识
开发Bluegiga APX4协议产品需掌握蓝牙技术、嵌入式系统开发、C语言编程、硬件设计及调试技能,熟悉Bluegiga API和相关开发工具。
|
3月前
|
算法 安全
【 第九章】软件设计师 之 多媒体基础
软件设计师 之 多媒体基础备考资料
【 第九章】软件设计师 之 多媒体基础
|
3月前
|
敏捷开发 设计模式 开发者
【揭秘终极利器】AgileEAS.NET:服务定位器模式的魔法,如何让企业级软件开发瞬间提速?揭秘背后的技术奥秘与实战指南!
【8月更文挑战第16天】AgileEAS.NET是基于DotNet的企业级敏捷开发平台,其服务定位器模式助力构建高度解耦系统。通过全局服务目录动态查找服务,避免硬编码依赖。在AgileEAS.NET中,服务定位器以静态类形式封装服务注册与检索功能。示例展示了如何注册与获取服务实例,如在`UserController`中通过服务定位器使用`IUserService`。此模式整合到框架生命周期管理,便于各处获取服务实例,提升开发效率。然而,应适度使用并考虑依赖注入容器以增强代码可维护性和可测试性。
69 4
|
3月前
|
C# Windows 监控
WPF应用跨界成长秘籍:深度揭秘如何与Windows服务完美交互,扩展功能无界限!
【8月更文挑战第31天】WPF(Windows Presentation Foundation)是 .NET 框架下的图形界面技术,具有丰富的界面设计和灵活的客户端功能。在某些场景下,WPF 应用需与 Windows 服务交互以实现后台任务处理、系统监控等功能。本文探讨了两者交互的方法,并通过示例代码展示了如何扩展 WPF 应用的功能。首先介绍了 Windows 服务的基础知识,然后阐述了创建 Windows 服务、设计通信接口及 WPF 客户端调用服务的具体步骤。通过合理的交互设计,WPF 应用可获得更强的后台处理能力和系统级操作权限,提升应用的整体性能。
98 0
|
6月前
|
设计模式 uml
【软件设计师备考 专题 】标准制订过程和各类标准的基本知识
【软件设计师备考 专题 】标准制订过程和各类标准的基本知识
77 0
|
存储 安全 搜索推荐
详解软件开发的标准过程(生命周期):跟着标准搞,设计没烦恼
详解软件开发的标准过程(生命周期):跟着标准搞,设计没烦恼
管理感悟:软件的特性
管理感悟:软件的特性
73 0
|
网络协议 测试技术 开发工具
游戏协议测试二:协议工具开发技术介绍
游戏客户端通常是与服务器直接进行通信,如何通过工具来进行协议的修改和创建呢?第一种就是直接在客户端代码里面增加一些类似GM指令的测试接口,来达到对每个接口的测试目的。第二种就是从外部对双方的通讯网络下手,通过第三方将协议截获修改后再发送给对方,从而达到协议测试的目的如图1,本篇文章就是针对第二种方式来做介绍。
1209 0
游戏协议测试二:协议工具开发技术介绍
|
索引
自然框架开发系列(一):自然框架 和 AgileEAS.NET 合作,开发b/s的药店系统!
  AgileEAS.NET平台开发Step By Step系列-药店系统-索引     看了药店系统系列,很敬佩作者,可以把自己的业务经验拿出来与大家分享,和作者魏琼东、CallHot 商量了一下,我们可以用药店系统的需求和业务逻辑,用自然框架做一个b/s的版本。
948 0
|
SQL 数据库 Windows
艾伟_转载:基于.NET平台的Windows编程实战(五)—— 问卷管理功能的实现
本系列文章导航 基于.NET平台的Windows编程实战(一)——前言 基于.NET平台的Windows编程实战(二)—— 需求分析与数据库设计 基于.NET平台的Windows编程实战(四)—— 数据库操作类的编写 基于.
1100 0
下一篇
无影云桌面