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类结构

        Android基于OpenCV实现QR二维码检测

        检测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()
           }
        }

        结果

        Android基于OpenCV实现QR二维码检测

        源码

        github.com/onlyloveyd/…

        来源:https://juejin.cn/post/6972280716922978311

        标签:Android,OpenCV,二维码
        0
        投稿

        猜你喜欢

      • C#语法之泛型的多种应用

        2022-03-15 02:02:43
      • Android实现仿360桌面悬浮清理内存

        2021-08-25 11:21:05
      • 安卓模拟器genymotion的安装与使用图文教程

        2021-12-13 19:28:36
      • Java 入门图形用户界面设计之单选按钮

        2023-08-29 13:40:02
      • IDEA2020.2.3中创建JavaWeb工程的完整步骤记录

        2022-03-15 05:16:02
      • Android 类似UC浏览器的效果:向上滑动地址栏隐藏功能

        2023-01-29 05:01:41
      • SpringCloud微服务基础简介

        2022-09-01 23:16:24
      • c# mutex互斥量的深入解析

        2022-03-13 02:38:42
      • c#图像截取实例

        2022-02-21 07:49:58
      • Android UI设计与开发之仿人人网V5.9.2最新版引导界面

        2022-10-10 17:34:16
      • Java过滤器doFilter里chain.doFilter()函数的理解

        2023-11-11 14:45:50
      • java UDP实现一个聊天工具的示例代码

        2021-09-19 18:41:47
      • 深入理解java虚拟机的故障处理工具

        2023-11-20 06:41:58
      • 详细了解C语言二叉树的建立与遍历

        2021-08-17 10:24:01
      • Android自定义弹框Dialog效果

        2022-09-21 02:40:41
      • opencv 做人脸识别 opencv 人脸匹配分析

        2023-07-09 06:34:44
      • Java中BigInteger与BigDecimal类用法总结

        2021-09-14 07:15:26
      • Java使用JDBC连接postgresql数据库示例

        2022-11-06 22:49:02
      • java实现在线预览--poi实现word、excel、ppt转html的方法

        2022-09-29 20:29:41
      • C#序列化与反序列化实例

        2023-05-05 21:05:27
      • asp之家 软件编程 m.aspxhome.com