android端使用openCV實(shí)現(xiàn)車牌檢測(cè)
現(xiàn)在,汽車的蹤影無(wú)處不在,公路上疾馳,大街邊臨停,小區(qū)中停靠,車庫(kù)里停泊。管理監(jiān)控如此龐大數(shù)量的汽車是個(gè)頭疼的問題。精明的人們把目光放在車牌上,因?yàn)檐嚺剖瞧嚨摹吧矸葑C”。所以車牌識(shí)別成為了焦點(diǎn),而車牌檢測(cè)是車牌識(shí)別的基礎(chǔ)和前提。本篇文章,主要討論使用openCV實(shí)現(xiàn)車牌檢測(cè)。
openCV是開源計(jì)算機(jī)視覺庫(kù),基于計(jì)算機(jī)視覺與機(jī)器學(xué)習(xí),提供強(qiáng)大的圖像處理能力。我們可以快速集成openCV庫(kù)到android端,其中一種方式是直接安裝openCV Manager,按需使用:?jiǎn)?dòng)服務(wù)去動(dòng)態(tài)加載。這樣前期配置更簡(jiǎn)單,但需要另外安裝一個(gè)APK。我更傾向另外一種方式:把依賴的module和動(dòng)態(tài)/靜態(tài)庫(kù)都導(dǎo)入Project。具體步驟如下:
1、導(dǎo)入module
先從官網(wǎng)下載openCVForAndroid的sdk,以3.2.0版本為例,找到依賴庫(kù)路徑,然后導(dǎo)入module。
2、導(dǎo)入動(dòng)態(tài)與靜態(tài)庫(kù)
在sdk里面找到lib目錄,把所有的.a和.so文件拷貝到項(xiàng)目的libs對(duì)應(yīng)ABI路徑下:
3、配置gradle
將依賴的靜態(tài)庫(kù)編譯到native-libs里面:
task nativeLibsToJar(type: Jar, description: 'create a jar archive of the native libs') { destinationDir file("$buildDir/native-libs") baseName 'native-libs' from fileTree(dir: 'libs', include: '**/*.so') into 'lib/' } tasks.withType(JavaCompile) { compileTask -> compileTask.dependsOn(nativeLibsToJar) } dependencies { compile fileTree(include: ['*.jar'], dir: 'libs') compile fileTree(dir: "$buildDir/native-libs", include: 'native-libs.jar') ...... }
好了,經(jīng)過(guò)配置三步曲,我們就可以愉快地使用openCV了。
------------------------中場(chǎng)休息---------------------------
接下來(lái)是調(diào)用三步曲:加載openCV、初始化車牌檢測(cè)器和執(zhí)行車牌檢測(cè)
1、加載openCV
調(diào)用openCVLoader去加載,如果加載成功進(jìn)行下一步操作:
private void initOpenCV(){ boolean result = OpenCVLoader.initDebug(); if(result){ Log.i(TAG, "initOpenCV success..."); //初始化車牌檢測(cè)器 mPlateDetector = new ObjectDetector(this, R.raw.haarcascade_license_plate, 3, new Scalar(255, 0, 0, 0)); mObject = new MatOfRect(); }else { Log.e(TAG, "initOpenCV fail..."); } }
2、初始化檢測(cè)器
使用車牌檢測(cè)的級(jí)聯(lián)分類xml文件進(jìn)行初始化:
/** * 創(chuàng)建級(jí)聯(lián)分類器 * @param context 上下文 * @param id 級(jí)聯(lián)分類器ID * @return 級(jí)聯(lián)分類器 */ private CascadeClassifier createDetector(Context context, int id) { CascadeClassifier javaDetector; InputStream is = null; FileOutputStream os = null; try { is = context.getResources().openRawResource(id); File cascadeDir = context.getDir(LICENSE_PLATE_MODEL, Context.MODE_PRIVATE); File cascadeFile = new File(cascadeDir, id + ".xml"); os = new FileOutputStream(cascadeFile); byte[] buffer = new byte[4096]; int bytesRead; while ((bytesRead = is.read(buffer)) != -1) { os.write(buffer, 0, bytesRead); } javaDetector = new CascadeClassifier(cascadeFile.getAbsolutePath()); if (javaDetector.empty()) { javaDetector = null; } boolean delete = cascadeDir.delete(); Log.i("ObjectDetector", "deleteResult=" + delete); return javaDetector; } catch (IOException e) { e.printStackTrace(); return null; } finally { try { if (null != is) { is.close(); } if (null != os) { os.close(); } } catch (IOException e) { e.printStackTrace(); } } }
3、執(zhí)行車牌檢測(cè)
由于openCV操作對(duì)象是Mat,所以我們得把Bitmap轉(zhuǎn)成Mat,然后轉(zhuǎn)成Gray灰度圖去進(jìn)行檢測(cè):
/** * 執(zhí)行車牌檢測(cè) * @param bitmap bitmap * @return 車牌檢測(cè)后的bitmap */ private Bitmap doPlateDetecting(Bitmap bitmap){ if(mPlateDetector != null && bitmap != null){ Mat mRgba = new Mat(); Mat mGray = new Mat(); //bitmap轉(zhuǎn)成map Utils.bitmapToMat(bitmap, mRgba); //rgba轉(zhuǎn)成灰度圖 Imgproc.cvtColor(mRgba, mGray, Imgproc.COLOR_RGBA2GRAY); // 檢測(cè)車牌 Rect[] object = mPlateDetector.detectObject(mGray, mObject); if(object != null && object.length > 0){ //檢測(cè)到車牌區(qū)域 Rect rect = object[0]; //矩形標(biāo)識(shí) Imgproc.rectangle(mRgba, rect.tl(), rect.br(), mPlateDetector.getRectColor(), 3); } //mat轉(zhuǎn)回bitmap Utils.matToBitmap(mRgba, bitmap); } return bitmap; }
其中,detectObject方法體是調(diào)用cascadeClassifier的detectMultiScale來(lái)完成檢測(cè)的:
public Rect[] detectObject(Mat gray, MatOfRect object) { mCascadeClassifier.detectMultiScale( gray, // 要檢查的灰度圖像 object, // 檢測(cè)到的車牌 1.1, // 表示在前后兩次相繼的掃描中,搜索窗口的比例系數(shù) mMinNeighbors, // 默認(rèn)是3 Objdetect.CASCADE_SCALE_IMAGE, getSize(gray, 80), // 檢測(cè)目標(biāo)最小值 getSize(gray, 800)); // 檢測(cè)目標(biāo)最大值 return object.toArray(); }
折騰了這么久,讓我們看看車牌檢測(cè)結(jié)果:
上面的車牌幾乎是水平的,那么傾斜的車牌能不能檢測(cè)到呢?真相就在下面:
角度發(fā)生傾斜的車牌也是可以檢測(cè)出來(lái),但是在后期的車牌識(shí)別,需要進(jìn)行傾斜校正。如果靜態(tài)檢測(cè)還不夠意思,那么請(qǐng)看動(dòng)態(tài)檢測(cè)的效果(轉(zhuǎn)換出來(lái)的gif有點(diǎn)模糊,各位莫怪):
接下來(lái)的一篇博客會(huì)與大家一起探討車牌識(shí)別,敬請(qǐng)期待。歡迎各位熱愛openCV與圖像處理的朋友提出建議,相互學(xué)習(xí)。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- Android基于OpenCV實(shí)現(xiàn)圖像修復(fù)
- Android OpenCv4 繪制多邊形的方法
- Android+OpenCv4實(shí)現(xiàn)邊緣檢測(cè)及輪廓繪制出圖像最大邊緣
- 如何在Android上使用opencv
- Android Studio4.0導(dǎo)入OpenCv4.3.0的方法步驟
- 使用Android Studio創(chuàng)建OpenCV4.1.0 項(xiàng)目的步驟
- Android+OpenCV4.2.0環(huán)境配置詳解(Android studio)
- OpenCV在Android上的應(yīng)用示例
- Android 通過(guò)cmake的方式接入opencv的方法步驟
- Android通過(guò)Java sdk的方式接入OpenCv的方法
- Android基于OpenCV實(shí)現(xiàn)非真實(shí)渲染
相關(guān)文章
android項(xiàng)目從Eclipse遷移到Android studio中常見問題解決方法
android項(xiàng)目從Eclipse遷移到Android studio中經(jīng)常會(huì)遇到一些問題,本文提供了Android studio使用中常見問題解決方法2018-03-03AndroidStudio 設(shè)置格式化斷行寬度教程
這篇文章主要介紹了AndroidStudio 設(shè)置格式化斷行寬度教程,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-03-03Android實(shí)現(xiàn)隨機(jī)圓形云標(biāo)簽效果
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)隨機(jī)圓形云標(biāo)簽效果,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-05-05總結(jié)Android App內(nèi)存優(yōu)化之圖片優(yōu)化
網(wǎng)上有很多大拿分享的關(guān)于Android性能優(yōu)化的文章,主要是通過(guò)各種工具分析,使用合理的技巧優(yōu)化APP的體驗(yàn),提升APP的流暢度,但關(guān)于內(nèi)存優(yōu)化的文章很少有看到。下面是我在實(shí)踐過(guò)程中使用的一些方法,很多都是不太成熟的項(xiàng)目,只是將其作為一種處理方式分享給大家。2016-08-08Android原生態(tài)實(shí)現(xiàn)分享轉(zhuǎn)發(fā)功能實(shí)例
大家好,本篇文章主要講的是Android原生態(tài)實(shí)現(xiàn)分享轉(zhuǎn)發(fā)功能實(shí)例,感興趣的同學(xué)趕快來(lái)看一看吧,對(duì)你有幫助的話記得收藏一下2021-12-12Android數(shù)據(jù)轉(zhuǎn)移之Launcher導(dǎo)出數(shù)據(jù)庫(kù)給另一臺(tái)機(jī)器加載
這篇文章主要介紹了Android系統(tǒng)中Launcher導(dǎo)出數(shù)據(jù)庫(kù)給另一臺(tái)機(jī)器加載的詳細(xì)流程,數(shù)據(jù)轉(zhuǎn)移是少見但早晚要用到的功能,感興趣的朋友快來(lái)提前掌握吧2021-11-11