电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁技術文章
文章詳情頁

Android基于OpenCV實現QR二維碼檢測

瀏覽:5日期:2022-09-18 08:23:02
目錄QR二維碼QR二維碼格式QR二維碼結構APIQRCodeDetector類結構檢測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個。【維基百科】

APIQRCodeDetector類結構

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/…

以上就是Android基于OpenCV實現QR二維碼檢測的詳細內容,更多關于Android OpenCV實現QR二維碼檢測的資料請關注好吧啦網其它相關文章!

標簽: Android
相關文章:
主站蜘蛛池模板: 温控器生产厂家-提供温度开关/热保护器定制与批发-惠州市华恺威电子科技有限公司 | 缠膜机|缠绕包装机|无纺布包装机-济南达伦特机械设备有限公司 | H型钢切割机,相贯线切割机,数控钻床,数控平面钻,钢结构设备,槽钢切割机,角钢切割机,翻转机,拼焊矫一体机 | HV全空气系统_杭州暖通公司—杭州斯培尔冷暖设备有限公司 | ◆大型吹塑加工|吹塑加工|吹塑代加工|吹塑加工厂|吹塑设备|滚塑加工|滚塑代加工-莱力奇塑业有限公司 | 生鲜配送系统-蔬菜食材配送管理系统-连锁餐饮订货配送软件-挪挪生鲜供应链管理软件 | 西安展台设计搭建_西安活动策划公司_西安会议会场布置_西安展厅设计西安旭阳展览展示 | 高防护蠕动泵-多通道灌装系统-高防护蠕动泵-www.bjhuiyufluid.com慧宇伟业(北京)流体设备有限公司 | 防渗土工膜|污水处理防渗膜|垃圾填埋场防渗膜-泰安佳路通工程材料有限公司 | 伶俐嫂培训学校_月嫂培训班在哪里报名学费是多少_月嫂免费政府培训中心推荐 | 高低温老化试验机-步入式/低温恒温恒湿试验机-百科 | 对辊破碎机-液压双辊式,强力双齿辊,四辊破碎机价格_巩义市金联机械设备生产厂家 | 储能预警-储能消防系统-电池舱自动灭火装置-四川千页科技股份有限公司官网 | 昆明网络公司|云南网络公司|昆明网站建设公司|昆明网页设计|云南网站制作|新媒体运营公司|APP开发|小程序研发|尽在昆明奥远科技有限公司 | 陕西华春网络科技股份有限公司 | 保定市泰宏机械制造厂-河北铸件厂-铸造厂-铸件加工-河北大件加工 | 武汉EPS线条_EPS装饰线条_EPS构件_湖北博欧EPS线条厂家 | 磁力反应釜,高压釜,实验室反应釜,高温高压反应釜-威海自控反应釜有限公司 | 一技任务网_有一技之长,就来技术任务网 | 上海质量认证办理中心| 南京交通事故律师-专打交通事故的南京律师 | 南京交通事故律师-专打交通事故的南京律师 | 传动滚筒_厂家-淄博海恒机械制造厂 | 禹城彩钢厂_钢结构板房_彩钢复合板-禹城泰瑞彩钢复合板加工厂 | 工业胀紧套_万向节联轴器_链条-规格齐全-型号选购-非标订做-厂家批发价格-上海乙谛精密机械有限公司 | 千淘酒店差旅平台-中国第一家针对TMC行业的酒店资源供应平台 | 洗地机-全自动/手推式洗地机-扫地车厂家_扬子清洁设备 | 杭州门窗厂家_阳光房_包阳台安装电话-杭州窗猫铝合金门窗 | 地脚螺栓_材质_标准-永年县德联地脚螺栓厂家 | 河南新乡德诚生产厂家主营震动筛,振动筛设备,筛机,塑料震动筛选机 | 高柔性拖链电缆-聚氨酯卷筒电缆-柔性屏蔽电缆厂家-玖泰电缆 | 拖链电缆_柔性电缆_伺服电缆_坦克链电缆-深圳市顺电工业电缆有限公司 | 艾默生变频器,艾默生ct,变频器,ct驱动器,广州艾默生变频器,供水专用变频器,风机变频器,电梯变频器,艾默生变频器代理-广州市盟雄贸易有限公司官方网站-艾默生变频器应用解决方案服务商 | EPDM密封胶条-EPDM密封垫片-EPDM生产厂家 | 会议会展活动拍摄_年会庆典演出跟拍_摄影摄像直播-艾木传媒 | 工业CT-无锡璟能智能仪器有限公司 | 喷漆房_废气处理设备-湖北天地鑫环保设备有限公司 | 耐腐蚀泵,耐腐蚀真空泵,玻璃钢真空泵-淄博华舜耐腐蚀真空泵有限公司 | 北京自然绿环境科技发展有限公司专业生产【洗车机_加油站洗车机-全自动洗车机】 | 首页-浙江橙树网络技术有限公司| 哲力实业_专注汽车涂料汽车漆研发生产_汽车漆|修补油漆品牌厂家 长沙一级消防工程公司_智能化弱电_机电安装_亮化工程专业施工承包_湖南公共安全工程有限公司 |