/   今日科技快讯   /

近日,为给老年人提供更便捷的出行服务,在交通运输部等主管部门的指导下,滴滴出行已在全国正式开通全国老年人电话叫车热线4006881700,为老年人提供电话叫车服务。同时,滴滴已于1月22日在全国上线“滴滴老年版”微信小程序。用户在小程序内可设置超市、学校、子女住址、医院等10个常用地址,老年人只需选择常用地址,即可实现“一键叫车”,目前小程序“一键叫车”可支持呼叫滴滴快车。

/   作者简介   /

本篇文章来自安和桥北的少年同学的投稿,和大家分享了二维码开发的相关功能讲解,相信会对大家有所帮助!同时也感谢作者贡献的精彩文章!

安和桥北的少年的博客地址:

https://juejin.cn/user/616168885849976

/   正文   /

一直为网上找的扫码项目无法快速且识别率高的扫描并解析出二维码而苦恼,zxing自己修改集成的难度又比较高(菜是原罪)。好在,经高人指导,接触到了谷歌的MLkit工程,实际体验下来,是真的好用,在此做一个经验分享,也是为了自己以后能记住。先来看看集成之后的效果图:

跟微信扫码比起来,不能说是一模一样,但至少是有点相似了。

如此迷你的二维码也能解析出来哦。接下来,我们来看下如何具体实现吧。

首先,来看下重中之重,MLKit的项目介绍吧。

https://developers.google.cn/ml-kit

可以看到,这个项目除了扫码功能,还有其他各种各样的好用且免费的SDK,像是文字识别,人脸识别等等,都可以依赖这个项目轻松搞定。无奈小弟才疏学浅,只接触了该项目的扫码功能,等来日水平提升了,定将其他的功能也一一体验一遍。

对了,这个项目除了Android端,还有IOS端的哦,感兴趣的IOS小伙伴可以自行参考

要使用这个项目,必不可少的,需要先引入工程。

dependencies {// ...// Use this dependency to bundle the model with your appimplementation 'com.google.mlkit:barcode-scanning:16.1.1'}

有google play条件的小伙伴,还可以选择添加google play的相关引用,甚至可以使用google play model。

dependencies {// ...// Use this dependency to use the dynamically downloaded model in Google Play Servicesimplementation 'com.google.android.gms:play-services-mlkit-barcode-scanning:16.1.4'}
<application ...>...<meta-dataandroid:name="com.google.mlkit.vision.DEPENDENCIES"android:value="barcode" /><!-- To use multiple models: android:value="barcode,model2,model3" --></application>

CameraX的支持当然也要一并加上。

// 版本号def camerax_version = "1.0.0-rc03"// 对camera 及 camera2的支持,可自行选择implementation "androidx.camera:camera-core:${camerax_version}"implementation "androidx.camera:camera-camera2:${camerax_version}"

项目使用的是基于jetpack lifecycle的框架,所以加上lifecycle的支持。

    implementation "androidx.camera:camera-lifecycle:${camerax_version}"

预览用到了CameraX自带的预览控件。

    implementation "androidx.camera:camera-view:1.0.0-alpha22"

接下来就是具体的代码实现了。首先,用到了相机及相册,自然需要添加相应的权限申请。

ActivityCompat.requestPermissions(this, arrayOf(Manifest.permission.CAMERA,Manifest.permission.WRITE_EXTERNAL_STORAGE),REQUEST_PERMISSION)

先来看看扫码界面的实现,主要包含了预览的PreviewView及用于绘制扫码线条及扫码结果的ScanOverlay。

<androidx.camera.view.PreviewViewandroid:id="@+id/previewView"android:layout_width="match_parent"android:layout_height="match_parent"/>
<com.hsmedia.mlkitdemo.ScanOverlayandroid:id="@+id/overlay"android:layout_width="match_parent"android:layout_height="0dp"app:layout_constraintTop_toBottomOf="@id/iv_exit"app:layout_constraintBottom_toTopOf="@id/tv_tips"android:layout_marginBottom="20dp"android:layout_marginTop="20dp"/>

接下来看一下如何开启CameraX的预览。CameraX自带了检测相机是否可用的监听,可以在相机可用之后,再进行后续操作。

cameraProviderFuture = ProcessCameraProvider.getInstance(this)
cameraProviderFuture.addListener(Runnable {val cameraProvider = cameraProviderFuture.get()bindScan(cameraProvider, overlay.width,overlay.height)
}, ContextCompat.getMainExecutor(this@BarcodeScanningActivity))

并且,CameraX绑定生命周期控件后,可以根据生命周期,自行释放相机,妈妈再也不用担心忘记关相机啦。

val preview : Preview = Preview.Builder().build()//绑定预览
preview.setSurfaceProvider(previewView.surfaceProvider)//使用后置相机
val cameraSelector : CameraSelector = CameraSelector.Builder().requireLensFacing(CameraSelector.LENS_FACING_BACK).build()
//将相机绑定到当前控件的生命周期
camera = cameraProvider.bindToLifecycle(this as LifecycleOwner, cameraSelector, imageAnalysis, preview)

这样,就可以开启预览啦,可以注意到,代码中绑定到生命周期的时候,使用了一个imageAnalysis的useCases,而这,就是用于图片扫描的组件了。

//配置图片扫描
val imageAnalysis = ImageAnalysis.Builder().setTargetResolution(Size(width, height)).setBackpressureStrategy(ImageAnalysis.STRATEGY_KEEP_ONLY_LATEST).build()

光这样还不行,还需要为图片扫描配置解析扫码内容的解析器QRCodeAnalyser,说了这么多,终于要用到MLKit的二维码解析了。

来看一下QRCodeAnalyser的具体实现。

@SuppressLint("UnsafeExperimentalUsageError")override fun analyze(imageProxy: ImageProxy) {val mediaImage = imageProxy.image ?: kotlin.run {imageProxy.close()return}
val image = InputImage.fromMediaImage(mediaImage, imageProxy.imageInfo.rotationDegrees)
detector.process(image).addOnSuccessListener { barCodes ->if (barCodes.size > 0){listener.invoke(barCodes[0],imageProxy.width,imageProxy.height)//接收到结果后,就关闭解析detector.close()}}.addOnFailureListener { Log.d(TAG, "Error: ${it.message}") }.addOnCompleteListener { imageProxy.close() }}

这个类主要实现了ImageAnalysis.Analyzer,并实现了analyze解析方法,其中imageProxy就是CameraX传递过来的图片扫描内容了。其主要内容,就是根据传递过来的图片扫描内容,使用detector进行解析,而这个detector又是怎么来的呢?

 //配置当前扫码格式
private val options = BarcodeScannerOptions.Builder().setBarcodeFormats(Barcode.FORMAT_QR_CODE,Barcode.FORMAT_AZTEC).build()
//获取解析器
private val detector = BarcodeScanning.getClient(options)

这里的BarcodeScanning就是MLKit提供的二维码解析组件了。解析完成后,对扫码结果进行后续操作。

//绑定图片扫描解析
imageAnalysis.setAnalyzer(Executors.newSingleThreadExecutor(), QRCodeAnalyser{barcode,imageWidth,imageHeight->//解绑当前所有相机操作cameraProvider.unbindAll()//初始化缩放比例initScale(imageWidth,imageHeight)barcode.boundingBox?.let {//扫描二维码的外边框矩形overlay.addRect(translateRect(it))Log.i(TAG, "bindScan: left:${it.left} right:${it.right} top:${it.top} bottom:${it.bottom}")}Handler().postDelayed({//延迟1S后返回结果val intent = Intent()intent.putExtra(SCAN_RESULT,barcode.rawValue)setResult(Activity.RESULT_OK, intent)},1000)
})

这里需要注意的是,因为手机上overlay的绘制区域与实际的图片扫描区域并不是一致的,所以需要对返回内容中二维码的外边框矩形进行一个转换,否则,最后绘制的结果点位置会出错。

首先,根据绘制区域的大小和图片扫描区域的大小,初始化缩放比例。

private fun initScale(imageWidth : Int, imageHeight : Int){if(isPortraitMode(this)){scaleY = overlay.height.toFloat() / imageWidth.toFloat()scaleX = overlay.width.toFloat() / imageHeight.toFloat()}else{scaleY = overlay.height.toFloat() / imageHeight.toFloat()scaleX = overlay.width.toFloat() / imageWidth.toFloat()}
}

这里需要注意的是,因为实际相机和扫描的角度,在竖屏模式下,有一个90度的差别,所以这里实际上做的比例是用绘制区域的高和扫描区域的宽,绘制区域的宽和扫描区域的高去做一个计算。

private fun translateX(x: Float): Float = x * scaleX
private fun translateY(y: Float): Float = y * scaleY//将扫描的矩形换算为当前屏幕大小
private fun translateRect(rect: Rect) = RectF(translateX(rect.left.toFloat()),translateY(rect.top.toFloat()),translateX(rect.right.toFloat()),translateY(rect.bottom.toFloat())
)

根据比例去计算实际的二维码外边框矩形,并计算出最后的绘制点,进行绘制。

最后的扫码效果,无论是扫码速度,还是解析率,我觉得都要比之前用的基于zxing的扫码工程要高,感兴趣的小伙伴可以自行尝试一下。

推荐阅读:

一文了解AQS(AbstractQueuedSynchronizer)

万字图文,带你学懂Handler和内存屏障

我和 Google 的代码差在哪里?

欢迎关注我的公众号

学习技术或投稿

长按上图,识别图中二维码即可关注

仿微信做个极速二维码扫描功能相关推荐

  1. android USB摄像头做条形码及二维码扫描(1)

    摘要: 1.前言 2.底层配置 3.JNI实现 4.总结 5.BUG及优化记录 android USB摄像头做条形码及二维码扫描(2) 1. 前言 公司做的产品基于android开发板搭建的控制系统, ...

  2. 基于MUI框架的使用HTML5+实现的二维码扫描功能

                                                                               Barcode的一个实现案例 一.简介 Barco ...

  3. Android实现二维码扫描功能(四)-ZXing识别图片二维码,相册选图

    简介 上一篇 Android实现二维码扫描功能(三)-闪光灯控制介绍了光线较弱情况下开启闪光灯来辅助二维码识别的方法. 本篇我们介绍如何识别相册中的图片(含二维码) 动态演示 使用模拟器录制了动画演示 ...

  4. JavaCV/OpenCV 二维码扫描功能

    JavaCV/OpenCV 二维码扫描功能 怎样配置工程就不再赘述,不清楚的读者可以网上查找资料,二维码扫描功能通过JavaCV实现起来还是挺简单的,主要OpenCV中QRCodeDetector提供 ...

  5. Android实现二维码扫描功能-ZXing识别图片二维码,相册选图

    文章目录 1.演示 2.权限问题 3.实现步骤 4.工具类 5.图片Uri处理(重要更新) 1.演示 2.权限问题 部分朋友在打开相册时遇到读写权限未授权的问题,我在开发的时候没有遇到,也没有注册读写 ...

  6. Android实现二维码扫描功能(一)ZXing插件接入

    简介 关于Android扫描二维码的功能实现,网上有很多相关资料.在对比之后,选用了前辈了修改过的ZXing直接接入到项目中,特制作此demo,介绍整个过程. (最新更新)本篇文章讲解的接入方法对部分 ...

  7. iOS 自带二维码扫描功能的实现

    #自从iOS7以后中新增了二维码扫描功能.因此可以在不借助第三方类库的情况下简单的写出二维码的扫描功能: 原生的二维码扫描功能在AVFoundation框架下,所以在使用原生的二维码扫描功能时要先导入 ...

  8. Flutter 3.X二维码扫描功能

    Flutter 3.X二维码扫描功能 1. pubspec.yaml文件添加依赖 2使用 3. 源代码 4.第二种方式 4.1 pubspec.yaml文件添加依赖 4.2 使用 源代码 1. pub ...

  9. 使用安卓实现一个二维码扫描功能(基于Android Studio)

    二维码扫描使用很广泛,在这里记录二维码扫描的功能,需要导入第三方的类库. 步骤一:在项目中导入第三方的类库,导入方法:往Android Studio中导入第三方类库文件这篇博客中有详细记录 步骤二:定 ...

最新文章

  1. 2021年大数据Spark(四):三种常见的运行模式
  2. 20162311 算法复杂度-3
  3. C++如何实现DNS域名解析转
  4. VC 2010的MFC函数,CMFCVisualManager::GetInstance()可能导致内存泄露
  5. 【前端必备】七、页面性能优化
  6. js 中时间格式化的几种方法
  7. POJ1107 ZOJ1042 UVALive2291 W's Cipher【密码+模拟】
  8. 基于 pureXML 技术的数据库表结构扩展
  9. 【python教程】-- 入门 | 小甲鱼《零基础入门学Python》教程笔记(知识点详细、源码可复制)全
  10. 数字电子技术基础阎石(第六版)基本公式和若干常用公式的电路证明方法
  11. JavaScript笔记-前端AES加密
  12. J2EE框架搭建大集合
  13. python计算闰年
  14. 用可视化解构BERT,我们从上亿参数中提取出了6种直观模式
  15. element tabs 的基础类型下划线不显示
  16. 表格一分为二html,如何在excel表中的将一个格子一分为二
  17. 1、junit学习之junit的基本介绍
  18. winUSB设备上位机驱动开发环境的搭建
  19. Cesium|xt3d模型展开动画
  20. Oh My ZSH让你的终端美如画

热门文章

  1. 实战 Java 第8天:开发商品详情查询接口
  2. 机器学习之泰坦尼克号预测生还案例的分析(逻辑回归)
  3. 增量备份与差异备份的区别(Incremental vs. differential backup: A comparison)
  4. 二维码怎么制作?手把手教你制作生成
  5. 计算机管理 位置不可用,Win10系统文件打不开提示位置不可用拒绝访问如何解决...
  6. 上拉电阻的作用原理_电容触摸屏原理以及敦泰TP FT5X06驱动
  7. art-pi lvgl添加触摸事件部分代码
  8. java.lang.NoSuchMethodError: No static method ..........
  9. v2rayN断网修复
  10. 图片社交php,图像社交时代