Android基于OpenCV实现QR二维码检测
作者:易冬 发布时间:2021-12-13 07:56:39
目录
QR二维码
QR二维码格式
QR二维码结构
API
QRCodeDetector类结构
检测QR二维码
识别QR二维码
检测并识别QR二维码
操作
结果
源码
QR二维码
QR码(英语:Quick Response Code;全称为快速响应矩阵图码)是二维码的一种,于1994年由日本DENSO WAVE公司发明。QR来自英文Quick Response的缩写,即快速反应,因为发明者希望QR码可以快速解码其内容。QR码使用四种标准化编码模式(数字、字母数字、字节(二进制)和日文(Shift_JIS))来存储数据。QR码常见于日本,为目前日本最通用的二维空间条码,在世界各国广泛运用于手机读码操作。QR码比普通一维条码具有快速读取和更大的存储资料容量,也无需要像一维条码般在扫描时需要直线对准扫描仪。因此其应用范围已经扩展到包括产品跟踪,物品识别,文档管理,库存营销等方面。【 * 】
QR二维码格式
QR码呈正方形,常见的是黑白两色。在3个角落,印有较小,像“回”字的正方图案。这3个是帮助解码软件定位的图案,用户不需要对准,无论以任何角度扫描,资料仍然可以正确被读取。日本QR码的标准JIS X 0510在1999年1月发布,而其对应的ISO国际标准ISO/IEC18004,则在2000年6月获得批准。根据Denso Wave公司的网站资料,QR码是属于开放式的标准,QR码的规格公开,虽由Denso Wave公司持有的专利权益,但不会被运行。除了标准的QR码之外,也存在一种称为“微型QR码”的格式,是QR码标准的缩小版本,主要是为了无法处理较大型扫描的应用而设计。微型QR码同样有多种标准,最高可存储35个字符。【 * 】
QR二维码结构
QR码最大特征为其左上,右上,左下三个大型的如同“回”字的黑白间同心方图案,为QR码识别定位标记,失去其中一个会影响识别。而呈棋盘般分布的有别与大定位标记的较小的同心方则为其校正标记,用于校正识别,版本1没有校正标记,版本2在右下方,其中心点在左下和右上定位标记的外边框的相交点,版本10开始以每个等距的方式出现在右下校正点至左下和右上定位标记的外边框的连线、左上与左下定位标记的外边框的连线、左上与右上定位标记的外边框的连线之间、这四边线上等距点对边相连线,版本10等距有1个,版本25为3个,版本40为5个。【 * 】
API
QRCodeDetector类结构
检测QR二维码
public boolean detect(Mat img, Mat points)
参数一:img,待检测是否含有QR二维码的的灰度图或者彩色(BGR)图像。
参数二:points,检测到的QR二维码的最小区域四边形的4个顶点坐标集合。
返回值:布尔类型,true,代表检测到QR二维码;false,代表未检测到QR二维码。
public boolean detectMulti(Mat img, Mat points)
参数一:img,待检测是否含有QR二维码的的灰度图或者彩色(BGR)图像。
参数二:points,多个检测结果QR二维码的最小区域四边形的4个顶点坐标集合。
返回值:布尔类型,true,代表检测到QR二维码;false,代表未检测到QR二维码。
识别QR二维码
public String decode(Mat img, Mat points, Mat straight_qrcode)
参数一:img,含有QR二维码的灰度图像或者彩色(BGR)图像。
参数二:points,detect方法得到的points值。数据量不可为空。
参数三:straight_qrcode,经过矫正和二值化的QR二维码。【可选参数】
返回值:字符串类型,如果解码失败,则为空串。
public boolean decodeMulti(Mat img, Mat points, List<String> decoded_info, List<Mat> straight_qrcode)
参数一:img,含有QR二维码的灰度图像或者彩色(BGR)图像。
参数二:points,detect方法得到的points值。数据量不可为空。
参数三:decoded_info,多个二维码的解码信息。
参数四:straight_qrcode,所有检测到的二维码矫正和二值化的后的结果集合。【可选参数】
返回值:布尔类型,true,代表解码成功,反之,解码失败。
检测并识别QR二维码
public String detectAndDecode(Mat img, Mat points, Mat straight_qrcode)
参数一:img,含有QR二维码的灰度图像或者彩色(BGR)图像。
参数二:points,检测到的QR二维码的最小区域四边形的4个顶点坐标。
参数三:straight_qrcode,经过矫正和二值化的QR二维码。【可选参数】
返回值:字符串类型,如果解码失败,则为空串。
public boolean detectAndDecodeMulti(Mat img, List<String> decoded_info, Mat points, List<Mat> straight_qrcode)
参数一:img,含有QR二维码的灰度图像或者彩色(BGR)图像。
参数二:decoded_info,多个二维码的解码信息。
参数三:points,检测到的多个QR二维码的最小区域四边形的4个顶点坐标集合。【可选参数】
参数四:straight_qrcode,所有检测到的二维码矫正和二值化的后的结果集合。【可选参数】
返回值:字符串类型,如果解码失败,则为空串。
操作
/**
* QR二维码检测
* author: yidong
* 2020/10/27
*/
class QRDetectActivity : AppCompatActivity() {
private lateinit var mBinding: ActivityQrDetectBinding
private lateinit var mQRCodeDetector: QRCodeDetector
private var mPhotoSavePath = ""
private lateinit var mUri: Uri
private lateinit var mSource: Mat
private lateinit var mGray: Mat
private lateinit var mOperationSheet: BottomSheetDialog
private lateinit var mSheetBinding: LayoutQrDetectOpBinding
private lateinit var mPhotoSheet: BottomSheetDialog
private lateinit var mPhotoOpBinding: LayoutPhotoOpBinding
// 请求相机权限
private val requestCameraPermission =
registerForActivityResult(ActivityResultContracts.RequestPermission()) {
if (it) {
mPhotoSavePath =
cacheDir.path + File.separator + "${System.currentTimeMillis()}.png"
mUri = MediaStoreUtils.getIntentUri(this, File(mPhotoSavePath))
requestCamera.launch(mUri)
} else {
Toast.makeText(applicationContext, "无相机权限", Toast.LENGTH_SHORT).show()
}
}
// 请求外部存储权限
private val requestStoragePermission =
registerForActivityResult(ActivityResultContracts.RequestPermission()) {
if (it) {
pickImage.launch("image/*")
} else {
Toast.makeText(applicationContext, "无存储权限", Toast.LENGTH_SHORT).show()
}
}
private val requestCamera = registerForActivityResult(ActivityResultContracts.TakePicture()) {
if (it) {
val bgr = Imgcodecs.imread(mPhotoSavePath, Imgcodecs.IMREAD_COLOR)
if (bgr.empty()) {
Toast.makeText(applicationContext, "读取拍照结果失败", Toast.LENGTH_SHORT).show()
return@registerForActivityResult
} else {
Imgproc.cvtColor(bgr, mSource, Imgproc.COLOR_BGR2RGB)
Imgproc.cvtColor(bgr, mGray, Imgproc.COLOR_BGR2GRAY)
mBinding.ivLena.showMat(mSource)
}
} else {
Toast.makeText(applicationContext, "拍照失败", Toast.LENGTH_SHORT).show()
}
}
private val pickImage = registerForActivityResult(ActivityResultContracts.GetContent()) {
if (it != null) {
val filePath = MediaStoreUtils.getMediaPath(this, it)
if (filePath.isNullOrEmpty()) {
Toast.makeText(applicationContext, "读取图片失败", Toast.LENGTH_SHORT).show()
return@registerForActivityResult
}
val bgr = Imgcodecs.imread(filePath, Imgcodecs.IMREAD_COLOR)
if (bgr.empty()) {
Toast.makeText(applicationContext, "读取图片失败", Toast.LENGTH_SHORT).show()
return@registerForActivityResult
} else {
Imgproc.cvtColor(bgr, mSource, Imgproc.COLOR_BGR2RGB)
Imgproc.cvtColor(bgr, mGray, Imgproc.COLOR_BGR2GRAY)
mBinding.ivLena.showMat(mSource)
}
} else {
Toast.makeText(applicationContext, "选图失败", Toast.LENGTH_SHORT).show()
}
}
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
mBinding = DataBindingUtil.setContentView(this, R.layout.activity_qr_detect)
mQRCodeDetector = QRCodeDetector()
mSource = Mat()
mGray = Mat()
val bgr = Utils.loadResource(this, R.drawable.qrcode)
Imgproc.cvtColor(bgr, mSource, Imgproc.COLOR_BGR2RGB)
Imgproc.cvtColor(bgr, mGray, Imgproc.COLOR_BGR2GRAY)
mBinding.ivLena.showMat(mSource)
createDialog()
}
private fun createDialog() {
mOperationSheet = BottomSheetDialog(this)
mSheetBinding = LayoutQrDetectOpBinding.inflate(layoutInflater, null, false)
mOperationSheet.setContentView(mSheetBinding.root)
mSheetBinding.tvDetect.setOnClickListener {
mOperationSheet.dismiss()
doDetect()
}
mSheetBinding.tvDecode.setOnClickListener {
mOperationSheet.dismiss()
doDecode()
}
mPhotoSheet = BottomSheetDialog(this)
mPhotoOpBinding = LayoutPhotoOpBinding.inflate(layoutInflater, null, false)
mPhotoSheet.setContentView(mPhotoOpBinding.root)
mPhotoOpBinding.tvCamera.setOnClickListener {
mPhotoSheet.dismiss()
requestCameraPermission.launch(
Manifest.permission.CAMERA
)
}
mPhotoOpBinding.tvPhoto.setOnClickListener {
mPhotoSheet.dismiss()
requestStoragePermission.launch(
Manifest.permission.WRITE_EXTERNAL_STORAGE
)
}
}
private fun doDetect() {
val points = Mat()
val isHasQr = mQRCodeDetector.detect(mSource, points)
if (isHasQr) {
val pointArr = FloatArray(8)
points.get(0, 0, pointArr)
Log.d(App.TAG, pointArr.toList().toString())
val tmp = mSource.clone()
for (i in pointArr.indices step 2) {
val start = Point(pointArr[i % 8].toDouble(), pointArr[(i + 1) % 8].toDouble())
val end = Point(pointArr[(i + 2) % 8].toDouble(), pointArr[(i + 3) % 8].toDouble())
Imgproc.line(tmp, start, end, Scalar(255.0, 0.0, 0.0), 8, Imgproc.LINE_8)
}
mBinding.ivResult.showMat(tmp)
tmp.release()
}
}
private fun doDecode() {
val points = Mat()
val isHasQr = mQRCodeDetector.detect(mGray, points)
if (isHasQr) {
val result = mQRCodeDetector.decode(mGray, points)
if (result.isEmpty()) {
Toast.makeText(applicationContext, "无法解码", Toast.LENGTH_SHORT).show()
} else {
Snackbar.make(mBinding.root, "解码结果:$result", 3000).show()
}
Log.d(App.TAG, result)
} else {
Toast.makeText(applicationContext, "未检测到QRCode", Toast.LENGTH_SHORT).show()
}
}
private fun selectMedia() {
if (this::mPhotoSheet.isInitialized) {
mPhotoSheet.show()
}
}
private fun selectOps() {
if (this::mOperationSheet.isInitialized) {
mOperationSheet.show()
}
}
override fun onCreateOptionsMenu(menu: Menu?): Boolean {
menuInflater.inflate(R.menu.menu_qr_detect, menu)
return true
}
override fun onOptionsItemSelected(item: MenuItem): Boolean {
when (item.itemId) {
R.id.menu_pick_photo -> selectMedia()
R.id.menu_qr_ops -> selectOps()
}
return true
}
override fun onDestroy() {
mSource.release()
mGray.release()
super.onDestroy()
}
}
结果
源码
github.com/onlyloveyd/…
来源:https://juejin.cn/post/6972280716922978311


猜你喜欢
- 本文实现Unity调用手机摄像,拍摄,然后识别二维码,显示二维码的内容。需要导入一个zxing.unity.dll文件,现在这个脚本的识别数
- 本文实例为大家分享了unity实现贪吃蛇游戏的具体代码,供大家参考,具体内容如下首先创建一个头部,编写脚本利用WASD控制头部的移动。Vec
- 绝大部分知识与实例来自O'REILLY的《Java网络编程》(Java Network Programming,Fourth Edi
- 一、前言微信接口调用验证最终需要用到的三个参数noncestr、timestamp、signature:接下来将会给出获取这三个参数的详细代
- 显示当前运行java代码的运行时的各种参数。不带显String操作。package systeminfo;import java.util.
- 闲来无事想玩玩双向通信,实现类似QQ的互发消息的功能。于是乎开始学习.Net Remoting..Net Remoting 是由客户端通过R
- 现象在日志配置文件 logback-spring.xml 中,无论怎么修改级别,mybatis 的 sql 日志都会打印出来。原因在 app
- 先看看效果图:源码下载:Android Navigation TabBar控件实现多彩标签栏代码:MainActivity.javapack
- 本文实例为大家分享了OpenGL绘制Bezier曲线的具体代码,供大家参考,具体内容如下最近在看Francis S Hill ,Jr 和 S
- 前言小伙伴们在使用C#开发时,可能需要将一些信息写入到txt,这里就给大家介绍几种常用的方法。方法:1.将由字符串组成的数组写入txt此种方
- Android 完全退出的实例详解首先,在基类BaseActivity里,注册RxBus监听:public class BaseActivi
- MainActivity如下: package cn.testcallback; import android.os.Bundle; imp
- 一、什么是IO流输入流和输出流。输入流:数据从数据源(文件)到程序(内存)的路径输出流:数据从程序(内存)到数据源(文件)的路径二、常用的文
- 轨迹压缩算法场景描述给定一个GPS数据记录文件,每条记录包含经度和维度两个坐标字段,根据距离阈值压缩记录,将过滤后的所有记录的经纬度坐标构成
- 题目:Binary Tree Maximum Path SumGiven a binary tree, find the maximum p
- android第一次新建项目是,相关依赖包需要下载很久,至少半小时,因为网速问题,还会多次下载失败。解决办法如下:1、通过镜像将gradle
- 跨域问题,其实百度上面有一堆的解决方案针对普通的情况其实百度上面的方案都是可行的。我这里主要介绍2种情况。当然我这里的配置都是基于网关的,而
- 新配置一个spring的MVC项目,发现对Get请求的中文参数出现了乱码:查看了SpingMVC中关于编码的配置(在web.xml中),如下
- C#连接本地.mdf文件:项目中右键点击,新增——数据——基于服务的数据库,项目下直接生成.mdf数据库文件,后台(数据库的写入用参数传递)
- 前端模板框架为Bootstrap,系统分为前台和后台。后台主要为管理员角色,功能有:商品类型管理、商品管理、订单管理、会员管理、管理员管理等