Android:身份证识别功能实现
说明:
此文使用华为SDK、百度SDK、百度在线API三种方式实现。
一、使用华为SDK实现身份证识别:
说明:免费,不需要联网。
1.AndroidManifest.xml添加权限:
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.CAMERA"/>
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/><uses-feature android:name="android.hardware.camera" />
<uses-feature android:name="android.hardware.camera.autofocus" />
2. 工程根目录build.gradle:
buildscript {repositories {//...省略之前配的其他地址maven { url "https://developer.huawei.com/repo/" }}dependencies {//...省略之前配的其他地址classpath 'com.huawei.agconnect:agcp:1.6.0.300'}
}allprojects {repositories {//...省略之前配的其他地址maven { url "https://developer.huawei.com/repo/" }}
}
3.工程/app/build.gradle:
//...省略之前配的其他地址
apply plugin: 'com.huawei.agconnect'
android {//...省略之前配的其他地址packagingOptions {exclude 'lib/arm64-v8a/libmsoptimize.so'}
}
dependencies {//...省略之前配的其他地址implementation 'com.huawei.hms:ml-computer-card-icr-cn:2.0.3.303'implementation 'com.huawei.hms:ml-computer-vision-ocr:2.0.5.301'implementation 'com.huawei.hms:ml-computer-vision-ocr-cn-model:2.0.5.301'implementation 'com.huawei.hms:ml-computer-vision-ocr-jk-model:2.0.5.301'implementation 'com.huawei.hms:ml-computer-vision-ocr-latin-model:2.0.5.301'implementation 'com.huawei.hms:ml-computer-card-gcr-plugin:2.0.1.301'
}
4.调用身份证SDK:
(1)身份证识别业务类:
public class HWIDCardScanBiz {private static HWIDCardScanBiz I;public static HWIDCardScanBiz I() {if (I == null) {I = new HWIDCardScanBiz();}return I;}/*** 打开身份证识别页,调用之前要申请蓝牙动态权限*/public void startIDCardScanActivity(Context context, IDCardScanCallback callback) {MLCnIcrCaptureConfig config = new MLCnIcrCaptureConfig.Factory().setFront(true).create();MLCnIcrCapture icrCapture = MLCnIcrCaptureFactory.getInstance().getIcrCapture(config);icrCapture.capture(new HWIDScanCallback(callback), context);}private class HWIDScanCallback implements MLCnIcrCapture.CallBack {private IDCardScanCallback callback;public HWIDScanCallback(IDCardScanCallback callback) {this.callback = callback;}/*** 识别成功回调*/@Overridepublic void onSuccess(MLCnIcrCaptureResult result) {if (result == null) {Toast.makeText(XApp.Companion.getContext(), "识别失败", Toast.LENGTH_SHORT);return;}if (callback != null) {IDCardInfo info = new IDCardInfo(result.name, result.sex, result.nation, result.birthday, result.address, result.idNum, result.authority, result.validDate, result.sideType);callback.onIDCardResult(info);}}@Overridepublic void onCanceled() {}@Overridepublic void onFailure(int retCode, Bitmap bitmap) {Toast.makeText(XApp.Companion.getContext(), "识别失败", Toast.LENGTH_SHORT);}@Overridepublic void onDenied() {}}public interface IDCardScanCallback {void onIDCardResult(IDCardInfo result);}
}
(2)调用识别业务类,更新UI:
HWIDCardScanBiz.I().startIDCardScanActivity(context, new HWIDCardScanBiz.IDCardScanCallback() {@Overridepublic void onIDCardResult(IDCardInfo result) {/*
result.name: 姓名
result.sex:性别
result.nation:国籍
result.birthday:生日
result.address:地址
result.idNum:身份证号
result.validDate:有效期*/}
});
二、使用百度云SDK实现身份证识别:
说明:免费额度个人帐户每月1千次/企业帐户每月2千次,要联网。
1.添加配置: (1)AndroidManifest.xml添加权限:
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.CAMERA"/>
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/><uses-feature android:name="android.hardware.camera" />
<uses-feature android:name="android.hardware.camera.autofocus" />
(2)proguard-rules.pro添加混淆例外:
-keep class com.baidu.ocr.sdk.**{*;}
-dontwarn com.baidu.ocr.**
(3)将ocrsdk.aar放入工程根\app\libs目录下。(4)将ocr_ui模块工程放入工程根目录下,在settings.gradle中添加导入模块工程的配置:
include ':app', 'ocrsdk', 'ocr_ui'
(5)工程/app/build.gradle,导入ocrsdk.aar与ocr_ui模块工程:
//...省略之前配的其他地址
dependencies {//...省略之前配的其他地址implementation(name: 'ocrsdk', ext: 'aar')implementation project(path: ':ocr_ui')
}
2.调用代码实现身份证识别: 第1步:调用SDK方法获取token; 第2步:调用SDK方法打开身份证识别界面; 第3步,在调用的Activity的onActivityResult中调用SDK识别图片中的身份信息。调用身份证SDK业务类:
public class BDIDCardScanBiz {public static final int REQUEST_CODE_CAMERA = 100;private boolean mHasGotToken;private static BDIDCardScanBiz I;public static BDIDCardScanBiz I() {if (I == null) {I = new BDIDCardScanBiz();}return I;}/*** 第1步:调用此方法获取token*/public void getAccessToken() {OCR.getInstance(App.getContext()).initAccessTokenWithAkSk(new OnResultListener() {@Overridepublic void onResult(Object o) {AccessToken accessToken = (AccessToken) o;String token = accessToken.getAccessToken();mHasGotToken = true;}@Overridepublic void onError(OCRError error) {// 调用失败,返回OCRError子类SDKError对象}}, App.getContext(), "Zxuz7GjLGsjBna44UjOQPVJv", "teLf4S7EjI5fjIshagZoovRSKlZSfPwM");}/*** 第2步:调用此方法打开身份证识别界面*/public void startIDCardScanActivity(Activity act) {if (!PermissionUtil.checkCameraPermission(act)) return;Intent intent = new Intent(act, CameraActivity.class);intent.putExtra(CameraActivity.KEY_OUTPUT_FILE_PATH,FileUtils.getSaveFile(App.getContext()).getAbsolutePath());intent.putExtra(CameraActivity.KEY_CONTENT_TYPE, CameraActivity.CONTENT_TYPE_ID_CARD_FRONT);act.startActivityForResult(intent, REQUEST_CODE_CAMERA);}/*** 第3步,在onActivityResult中调此方法,获取身份证信息*/public void handlerData(String contentType) {if (TextUtils.isEmpty(contentType)) return;String filePath = FileUtils.getSaveFile(App.getContext()).getAbsolutePath();if (CameraActivity.CONTENT_TYPE_ID_CARD_FRONT.equals(contentType)) {recIDCard(IDCardParams.ID_CARD_SIDE_FRONT, filePath);} else if (CameraActivity.CONTENT_TYPE_ID_CARD_BACK.equals(contentType)) {recIDCard(IDCardParams.ID_CARD_SIDE_BACK, filePath);}}private void recIDCard(String idCardSide, String filePath) {IDCardParams param = new IDCardParams();param.setImageFile(new File(filePath));// 设置身份证正反面param.setIdCardSide(idCardSide);// 设置方向检测param.setDetectDirection(true);// 设置图像参数压缩质量0-100, 越大图像质量越好但是请求时间越长。 不设置则默认值为20param.setImageQuality(20);param.setDetectRisk(true);OCR.getInstance(App.getContext()).recognizeIDCard(param, new OnResultListener<IDCardResult>() {@Overridepublic void onResult(IDCardResult result) {if (result != null) {//获取身份信息}}@Overridepublic void onError(OCRError error) {}});}
}
三、使用百度云API实现身份证识别:
说明:免费额度个人帐户每月1千次/企业帐户每月2千次,要联网。
1.AndroidManifest.xml添加权限:
同上面两种
2.调用代码实现身份证识别:
第1步,调用百度API获取token:
public static String getToken() {String url = "https://aip.baidubce.com/oauth/2.0/token?client_id=" + Config.CLIENT_ID + "&client_secret=" + Config.CLIENT_SECRET + "&grant_type=client_credentials";if (!NetHelp.getCurConnectStatus()) {return null;}Response response = null;try {Request request = new Request.Builder().url(url).get().build();LogUtils.d("getToken request url :" + url);response = httpClient.newCall(request).execute();} catch (IOException e) {LogUtils.d("getToken request exception: " + e.getMessage());e.printStackTrace();return null;}LogUtils.d("getToken response isSuccessful: " + response.isSuccessful());if (response != null && response.isSuccessful()) {try {String strRecData = response.body().string();LogUtils.d("getToken response body: " + strRecData);if (strRecData != null && strRecData.length() > 2) {JSONObject jsonObj = new JSONObject(strRecData);if (jsonObj != null && !jsonObj.isNull("access_token")) {return jsonObj.getString("access_token");}}return response.body().string();} catch (Exception e) {LogUtils.d("getToken response parse exception: " + e.getMessage());e.printStackTrace();return null;}}return null;}
第2步,调用系统相机拍身份证照片(可以自已实现带身份证头像框的拍照功能,界面更加美观):
/*** 调起系统相机进行拍照,此步可以自已实现拍照功能,界面更美观*/public void startIDCardScanActivity(Activity act) {Log.i("IDCard", "IDCard startIDCardScanActivity >>>");Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);if (intent.resolveActivity(act.getPackageManager()) != null) {act.startActivityForResult(intent, REQUEST_CODE_CAMERA);}}
第3步,在onActivityResult中,将图片上传百度API获取身份证信息:
/*** 解析身份证图片,获取身份信息*/public void handlerData(Activity act, Intent data, IDCardScanCallback callback) {new Thread() {@Overridepublic void run() {Bundle bundle = data.getExtras();Log.i("IDCard", "IDCard handlerData bundle >>> : " + bundle);if (bundle == null) return;Bitmap bitmap = (Bitmap) bundle.get("data");Log.i("IDCard", "IDCard handlerData bitmap >>> : " + bitmap);if (bitmap == null) return;File file = new File(App.getContext().getCacheDir(), System.currentTimeMillis() + ".jpeg");if (!file.getParentFile().exists()) file.mkdirs();try {FileOutputStream out = new FileOutputStream(file);bitmap.compress(Bitmap.CompressFormat.JPEG, 80, out);out.flush();out.close();Log.i("IDCard", "IDCard saveBitmap filePath: " + file.getAbsolutePath() + " fileLen: " + file.length());} catch (IOException ex) {ex.printStackTrace();}String filePath = file.getAbsolutePath();Log.i("IDCard", "IDCard parseImage >>> filePath: " + filePath);final IDCardInfo info = HttpHelp.getIDCard(new File(filePath));if (info != null && callback != null) {act.runOnUiThread(() -> callback.onIDCardResult(info));}}}.start();}
public static IDCardInfo getIDCard(File file){if (!NetHelp.getCurConnectStatus() || file == null || file.length() <= 0){return null;}String accessToken = getToken();if (TextUtils.isEmpty(accessToken)) return null;String url = "https://aip.baidubce.com/rest/2.0/ocr/v1/idcard?access_token=" + accessToken;String img = null;try {img = FileUtils.base64File(file);} catch (Exception e) {e.printStackTrace();}if (TextUtils.isEmpty(img)) return null;Response response = null;try {FormBody.Builder builder = new FormBody.Builder().add("id_card_side", "front").add("image", img);Request request = new Request.Builder().url(url).post(builder.build()).build();LogUtils.d("getIDCard request url :" + url + " img: " + img);response = httpClient.newCall(request).execute();} catch (IOException e) {LogUtils.d("getIDCard request exception: " + e.getMessage());e.printStackTrace();return null;}LogUtils.d("getIDCard response isSuccessful: " + response.isSuccessful());if (response != null && response.isSuccessful()) {try {String strRecData = response.body().string();LogUtils.d("getIDCard response body: " + strRecData);if (strRecData != null && strRecData.length() > 2) {JSONObject rootObj = new JSONObject(strRecData);if (rootObj != null && !rootObj.isNull("words_result")) {JSONObject wordObj = rootObj.getJSONObject("words_result");IDCardInfo info = new IDCardInfo();if (!wordObj.isNull("姓名") && !wordObj.getJSONObject("姓名").isNull("words")) {info.name = wordObj.getJSONObject("姓名").getString("words");}if (wordObj != null && !wordObj.isNull("民族") && !wordObj.getJSONObject("民族").isNull("words")) {info.nation = wordObj.getJSONObject("民族").getString("words");}if (wordObj != null && !wordObj.isNull("住址") && !wordObj.getJSONObject("住址").isNull("words")) {info.address = wordObj.getJSONObject("住址").getString("words");}if (wordObj != null && !wordObj.isNull("公民身份号码") && !wordObj.getJSONObject("公民身份号码").isNull("words")) {info.idNum = wordObj.getJSONObject("公民身份号码").getString("words");}if (wordObj != null && !wordObj.isNull("出生") && !wordObj.getJSONObject("出生").isNull("words")) {info.birthday = wordObj.getJSONObject("出生").getString("words");}if (wordObj != null && !wordObj.isNull("性别") && !wordObj.getJSONObject("性别").isNull("words")) {info.sex = wordObj.getJSONObject("性别").getString("words");}LogUtils.d("getIDCard response parse IDCardInfo: " + info.toString());return info;}}} catch (Exception e) {LogUtils.d("getIDCard response parse exception: " + e.getMessage());e.printStackTrace();return null;}}return null;}
相关文章:
Android:身份证识别功能实现
说明: 此文使用华为SDK、百度SDK、百度在线API三种方式实现。 一、使用华为SDK实现身份证识别: 说明:免费,不需要联网。 1.AndroidManifest.xml添加权限:<uses-permission android:name"android.permissio…...
MacOS安装Homebrew教程
安装 Homebrew 是在 macOS 上管理软件包的一种简便方法。以下是安装 Homebrew 的步骤: 打开终端:你可以通过在 Spotlight 搜索栏中输入“终端”并按下回车键来打开 macOS 的终端应用程序。 执行安装命令:在终端中粘贴以下命令并按下回车键执…...
laravel如何通过DB获取一条数据并转成数组
在 Laravel 中,你可以使用原生数据库查询构建器(DB facade)来获取一条数据,并将其转换为数组。这可以通过在查询链的末尾调用 first() 方法后,使用 toArray() 方法来实现。first() 方法会返回一个 StdClass 对象&#…...

ENSP USG防火墙接入虚拟机;开启Web访问;
1.添加防火墙及云,启动防火墙; 2.配置桥接网卡; 默认账户:admin 默认密码:Admin123 #第一次登陆需修改密码; 默认G0/0/0口为管理口,而在模拟器中进入防火墙的web需如下配置: 配置 …...

数据结构算法题(力扣)——链表
以下题目建议大家先自己动手练习,再看题解代码。这里只提供一种做法,可能不是最优解。 1. 移除链表元素(OJ链接) 题目描述:给一个链表的头节点 head 和一个整数 val ,删除链表中所有满足值等于 val 的节点…...

LeetCode---391周赛
题目列表 3099. 哈沙德数 3100. 换水问题 II 3101. 交替子数组计数 3102. 最小化曼哈顿距离 一、哈沙德数 简单的模拟题,代码如下 class Solution { public:int sumOfTheDigitsOfHarshadNumber(int x) {int s 0, tmp x;while(tmp){stmp%10;tmp/10;}return x…...

微信小程序的页面交互2
一、自定义属性 (1)定义: 微信小程序中的自定义属性实际上是由data-前缀加上一个自定义属性名组成。 (2)如何获取自定义属性的值? 用到target或currentTarget对象的dataset属性可以获取数据 ÿ…...

【VSCode】修改插件地址
不想放在原始C盘下面C:\Users\{用户}\.vscode\extensions为了后续存储空间考虑,想通过添加环境变量创建名为VSCODE_EXTENSIONS的环境变量,内容指向vs Code扩展所在目录即可 直接配置环境变量,不要在有空格的文件夹下面 变量名称:…...

自然语言处理NLP概述
大家好,自然语言处理(NLP)是计算机科学领域与人工智能领域中的一个重要方向,其研究能实现人与计算机之间用自然语言进行有效通信的各种理论和方法。本文将从自然语言处理的本质、原理和应用三个方面,对其进行概述。 一、NLP的本质 NLP是一种…...

计算机网络——37认证
认证 目标:Bob需要Alice证明他的身份 Protocol ap1.0:Alice说"A am Alice" 可能出现的问题: 在网络上Bob看不到Alice,因此Trudy可以简单的声称他是Alice 认证:重新尝试 Protocol ap2.0:Alice…...

Java中利用BitMap位图实现海量级数据去重
🏷️个人主页:牵着猫散步的鼠鼠 🏷️系列专栏:Java全栈-专栏 🏷️个人学习笔记,若有缺误,欢迎评论区指正 目录 前言 什么是BitMap?有什么用? 基本概念 位图的优势 …...
Linux知识点记录
Linux知识点记录 1. 后台运行应用程序方法一:&方法二:nohup & 2. 一个shell脚本中执行多个应用程序3. 2>&14. shell脚本清除日志5. 通过grep查找匹配字符串 1. 后台运行应用程序 参考文章:https://blog.csdn.net/Pan_peter/…...
js的check函数
在JavaScript中,并没有一个内置的名为check的函数。然而,你可以根据需求自定义一个check函数,用于执行各种验证和检查任务。这个check函数的具体作用完全取决于你如何定义和实现它。 以下是一个简单的示例,展示了如何定义一个che…...

赛尼格磁电科技邀您到场参观2024第13届生物发酵展
参展企业介绍 北京赛尼格磁电科技有限公司是一家中加合资的专业永磁组件生产商,2001年成立于中国北京。公司专业从事磁性材料的应用及各类磁系统的设计、开发及制造,公司产品广泛应用于汽车行业、建筑行业、电子行业、航海领域、医学领域、教育领域等。 …...

gpt国内怎么用?最新版本来了
claude 3 opus面世后,这几天已经有许多应用,而其精确以及从不偷懒(截止到2024年3月11日还没有偷懒)的个性,也使得我们可以用它来首次完成各种需要多轮对话的尝试。 今天我们想要进行的一项尝试就是—— 如何从一个不知…...
Vim脚本语言入门:打造你的编辑器
简介 Vim脚本语言是Vim编辑器内置的一种脚本语言,它赋予用户高度的定制和自动化编辑任务的能力。通过编写Vim脚本,用户可以根据自己的需求来扩展和改进Vim编辑器的功能,从而提高编辑效率和舒适度。 在Vim中,脚本语言被广泛用于创…...
myweb项目资料集
项目要求 前后端分离后端采用 flask 框架前端采用 vue3 框架 后端部分 Flask 3 框架: https://dormousehole.readthedocs.io/en/latest/quickstart.html Session: https://blog.csdn.net/zhangvalue/article/details/93892241 MySQL 操作…...

Kubernetes(k8s):部署、使用 metrics-server
Kubernetes(k8s):部署、使用 metrics-server 一、metrics-server简介二、部署metrics-server2.1、 下载 Metrics Server 部署文件2.2、修改metrics-server.yaml 文件2.3、 部署 Metrics Server2.4、 检查 Metrics Server 三、使用 Metrics Se…...
为什么建议你学习Spring底层原理?
1.根因 Java诞生以来,一直是业界的主流语言和平台,而Spring则是Java开发的平台。与其说是用Java编程,不如说是在Spring框架上编程。即便最近几年比较火的Spring Boot、Spring Cloud,其底层内核仍然是Spring。因此,作为…...

post请求搜索功能爬虫
<!--爬虫仅支持1.8版本的jdk--> <!-- 爬虫需要的依赖--> <dependency> <groupId>org.apache.httpcomponents</groupId> <artifactId>httpclient</artifactId> <version>4.5.2</version> </dependency>…...
利用ngx_stream_return_module构建简易 TCP/UDP 响应网关
一、模块概述 ngx_stream_return_module 提供了一个极简的指令: return <value>;在收到客户端连接后,立即将 <value> 写回并关闭连接。<value> 支持内嵌文本和内置变量(如 $time_iso8601、$remote_addr 等)&a…...

Spark 之 入门讲解详细版(1)
1、简介 1.1 Spark简介 Spark是加州大学伯克利分校AMP实验室(Algorithms, Machines, and People Lab)开发通用内存并行计算框架。Spark在2013年6月进入Apache成为孵化项目,8个月后成为Apache顶级项目,速度之快足见过人之处&…...

【入坑系列】TiDB 强制索引在不同库下不生效问题
文章目录 背景SQL 优化情况线上SQL运行情况分析怀疑1:执行计划绑定问题?尝试:SHOW WARNINGS 查看警告探索 TiDB 的 USE_INDEX 写法Hint 不生效问题排查解决参考背景 项目中使用 TiDB 数据库,并对 SQL 进行优化了,添加了强制索引。 UAT 环境已经生效,但 PROD 环境强制索…...

从零开始打造 OpenSTLinux 6.6 Yocto 系统(基于STM32CubeMX)(九)
设备树移植 和uboot设备树修改的内容同步到kernel将设备树stm32mp157d-stm32mp157daa1-mx.dts复制到内核源码目录下 源码修改及编译 修改arch/arm/boot/dts/st/Makefile,新增设备树编译 stm32mp157f-ev1-m4-examples.dtb \stm32mp157d-stm32mp157daa1-mx.dtb修改…...

2025盘古石杯决赛【手机取证】
前言 第三届盘古石杯国际电子数据取证大赛决赛 最后一题没有解出来,实在找不到,希望有大佬教一下我。 还有就会议时间,我感觉不是图片时间,因为在电脑看到是其他时间用老会议系统开的会。 手机取证 1、分析鸿蒙手机检材&#x…...

基于SpringBoot在线拍卖系统的设计和实现
摘 要 随着社会的发展,社会的各行各业都在利用信息化时代的优势。计算机的优势和普及使得各种信息系统的开发成为必需。 在线拍卖系统,主要的模块包括管理员;首页、个人中心、用户管理、商品类型管理、拍卖商品管理、历史竞拍管理、竞拍订单…...

代码规范和架构【立芯理论一】(2025.06.08)
1、代码规范的目标 代码简洁精炼、美观,可持续性好高效率高复用,可移植性好高内聚,低耦合没有冗余规范性,代码有规可循,可以看出自己当时的思考过程特殊排版,特殊语法,特殊指令,必须…...
Python网页自动化Selenium中文文档
1. 安装 1.1. 安装 Selenium Python bindings 提供了一个简单的API,让你使用Selenium WebDriver来编写功能/校验测试。 通过Selenium Python的API,你可以非常直观的使用Selenium WebDriver的所有功能。 Selenium Python bindings 使用非常简洁方便的A…...

高保真组件库:开关
一:制作关状态 拖入一个矩形作为关闭的底色:44 x 22,填充灰色CCCCCC,圆角23,边框宽度0,文本为”关“,右对齐,边距2,2,6,2,文本颜色白色FFFFFF。 拖拽一个椭圆,尺寸18 x 18,边框为0。3. 全选转为动态面板状态1命名为”关“。 二:制作开状态 复制关状态并命名为”开…...

性能优化中,多面体模型基本原理
1)多面体编译技术是一种基于多面体模型的程序分析和优化技术,它将程序 中的语句实例、访问关系、依赖关系和调度等信息映射到多维空间中的几何对 象,通过对这些几何对象进行几何操作和线性代数计算来进行程序的分析和优 化。 其中࿰…...