OpenCV 在 Android 上的应用

Posted Java与Android技术栈

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了OpenCV 在 Android 上的应用相关的知识,希望对你有一定的参考价值。

一. OpenCV 介绍

OpenCV是一个基于BSD许可(开源)发行的跨平台计算机视觉库,可以运行在Linux、Windows、android和Mac OS操作系统上。它轻量级而且高效——由一系列 C 函数和少量 C++ 类构成,同时提供了Python、Ruby、MATLAB等语言的接口,实现了图像处理和计算机视觉方面的很多通用算法。

在移动端上使用 OpenCV 可以完成一系列图像处理的工作。

二. OpenCV 在 Android 上的配置

我在项目中使用的 OpenCV 版本是 4.x。

在 Android Studio 中创建一个 Library,将官网下载的 OpenCV 导入后,就可以直接调用 OpenCV 中 Java 类的方法。

如果想调用 C++ 的类,也可以使用 CMake 创建环境,然后通过 include 文件放入指定路径。

下面是项目中使用的 CMakeLists.txt

 
   
   
 
  1. cmake_minimum_required(VERSION 3.6.0)


  2. include_directories(

  3. ${CMAKE_SOURCE_DIR}/src/main/cpp/include

  4. )


  5. add_library(libopencv_java4 SHARED IMPORTED)

  6. set_target_properties(

  7. libopencv_java4

  8. PROPERTIES IMPORTED_LOCATION

  9. ${CMAKE_SOURCE_DIR}/src/main/jniLibs/libs/${ANDROID_ABI}/libopencv_java4.so)


  10. add_library(libc++_shared SHARED IMPORTED)

  11. set_target_properties(

  12. libc++_shared

  13. PROPERTIES IMPORTED_LOCATION

  14. ${CMAKE_SOURCE_DIR}/src/main/jniLibs/libs/${ANDROID_ABI}/libc++_shared.so)



  15. add_library(

  16. detect


  17. SHARED


  18. src/main/cpp/detect-lib.cpp

  19. src/main/cpp/detect-phone.cpp

  20. )



  21. find_library(

  22. log-lib

  23. log

  24. )


  25. target_link_libraries(

  26. detect libopencv_java4 libc++_shared jnigraphics

  27. ${log-lib}

  28. )

其中,detect-lib.cpp 和 detect-phone.cpp 是我创建的 C++ 类。打成 so 文件时,会包含这2个类。

三. 例子两则

3.1 作为二维码识别的兜底方案

在 Android 原生开发中,二维码识别有老牌的 zxing 等开源库。为何还要使用 OpenCV 呢?

下面的代码,展示了在应用层拍完照之后,将图片的路径传到 jni 层将其转换成对应的 Mat 对象,再转换成灰度图像,然后找出二维码的位置,要是能够找到的话就识别出二维码的内容。

 
   
   
 
  1. extern "C"

  2. JNIEXPORT jstring JNICALL

  3. Java_com_xxx_sdk_utils_DetectUtils_qrDetect(JNIEnv *env, jclass jc,jstring filePath) {


  4. const char *file_path_str = env->GetStringUTFChars(filePath, 0);

  5. string path = file_path_str;

  6. Mat src = imread(path);


  7. Mat gray, qrcode_roi;

  8. cvtColor(src, gray, COLOR_BGR2GRAY);

  9. QRCodeDetector qrcode_detector;

  10. vector<Point> pts;

  11. string detect_info;

  12. bool det_result = qrcode_detector.detect(gray, pts);

  13. if (det_result) {

  14. detect_info = qrcode_detector.decode(gray, pts, qrcode_roi);

  15. return env->NewStringUTF(detect_info.c_str());

  16. } else {

  17. detect_info = "";

  18. return env->NewStringUTF(detect_info.c_str());

  19. }

  20. }

对应的 Java 代码,方便应用层调用 jni 层的 qrDetect()

 
   
   
 
  1. public class DetectUtils {


  2. static {

  3. System.loadLibrary("detect");

  4. }


  5. /**

  6. * @param filePath

  7. * @return

  8. */

  9. public static native String qrDetect(String filePath);


  10. ......

  11. }

最后是应用层的调用

 
   
   
 
  1. // 使用 OpenCV 进行二维码识别

  2. val result = DetectUtils.qrDetect(filePath)

3.2 比对图像的差异

在我们的实际开发中遇到一个应用场景:需要判断我们的手机回收机里面是否存放了物体。(手机回收机是一个触摸屏设备,可以通过 Android 系统来操作内部的硬件设备。)

我们事先拍一张回收机内没有物体的图作为基准图像,等到需要判断是否存在物体时再拍一张图片。两幅图片对比看比例,比列超过阈值则认为回收机内存在着物体。

下面的代码,展示了在应用层拍完照之后,跟基准图片进行比对,并返回结果。

 
   
   
 
  1. extern "C"

  2. JNIEXPORT jboolean JNICALL

  3. Java_com_xxx_sdk_utils_DetectUtils_checkPhoneInMTA(JNIEnv *env, jclass jc,jstring baseImgPath,jstring filePath) {


  4. jboolean tRet = false;

  5. const char *file_path_str = env->GetStringUTFChars(filePath, 0);

  6. string path = file_path_str;

  7. Mat src = imread(path);


  8. const char *base_img_path_str = env->GetStringUTFChars(baseImgPath, 0);

  9. string basePath = base_img_path_str;

  10. Mat baseImg = imread(basePath);


  11. int result = checkPhoneInBox(baseImg,src,40,0.1);


  12. LOGI("checkPhoneInBox result = %d",result);

  13. if (result == 0) {

  14. tRet = true;

  15. }


  16. return tRet;

  17. }

两张图片真正的比对是在 checkPhoneInBox() 中完成的。其中,maxFilter() 是为了处理彩色的情况,然后使用高斯滤波进行降噪处理,再进行二值化处理,最后判断灰度差异区域占总图像的比列是否超过预先设定的阈值。

 
   
   
 
  1. int checkPhoneInBox(cv::Mat baseImg, cv::Mat snapImg, int diffThresh, double threshRatio) {


  2. cv::Mat baseMaxImg, snapMaxImg,baseGausImg, snapGausImg;

  3. if (baseImg.empty()|| snapImg.empty())

  4. {

  5. return -1;

  6. }


  7. try {

  8. maxFilter(baseImg, baseMaxImg);

  9. maxFilter(snapImg, snapMaxImg);

  10. } catch (...) {

  11. return -1;

  12. }


  13. cv::GaussianBlur(baseMaxImg, baseGausImg, cv::Size(5, 5),0);

  14. cv::GaussianBlur(snapMaxImg, snapGausImg, cv::Size(5, 5),0);


  15. cv::Mat diff,diffBin;

  16. cv::Mat noMax;

  17. cv::absdiff(baseGausImg, snapGausImg, diff);

  18. cv::threshold(diff, diffBin, diffThresh, 255, cv::THRESH_BINARY);


  19. float ratio = (float)cv::countNonZero(diffBin) / (long)diffBin.total();


  20. LOGI("ratio = %f,%d,%ld",ratio,cv::countNonZero(diffBin),(long)diffBin.total());


  21. if (ratio > threshRatio)

  22. {

  23. return 0;

  24. }

  25. else

  26. {

  27. return 1;

  28. }

  29. }


  30. int maxFilter(cv::Mat baseImg, cv::Mat &maxImg)

  31. {

  32. if (baseImg.channels() <3)

  33. {

  34. maxImg = baseImg.clone();

  35. }

  36. else

  37. {

  38. maxImg.create(baseImg.size(), CV_8UC1);

  39. for (int r=0;r<baseImg.rows;r++)

  40. {

  41. for (int c = 0; c < baseImg.cols; c++)

  42. {

  43. uchar maxTmp=0;

  44. cv::Vec3b s = baseImg.at<cv::Vec3b>(r, c);

  45. maxTmp = (std::max)(s[0],s[1]);

  46. maxTmp = (std::max)(maxTmp,s[2]);


  47. maxImg.at<uchar>(r, c) = maxTmp;

  48. }

  49. }

  50. }

  51. return 0;

  52. }

对应的 Java 代码,方便应用层调用 jni 层的 checkPhoneInMTA()

 
   
   
 
  1. public class DetectUtils {


  2. static {

  3. System.loadLibrary("detect");

  4. }


  5. /**

  6. * 判断MTA中是否有手机

  7. * @param baseImageFilePath 基准的图片

  8. * @param filePath 拍摄的图片

  9. * @return

  10. */

  11. public static native boolean checkPhoneInMTA(String baseImageFilePath, String filePath);


  12. ......

  13. }

最后是应用层的调用

 
   
   
 
  1. val result = DetectUtils.checkPhoneInMTA(Constants.OPENCV_PHOTO_PATH, it.absolutePath)

四. 总结

OpenCV 是一款功能强大的图像处理库。但是它本身体积也较大,在移动端使用至少会增加 Android Apk 包 10 M+ 的体积(主要取决于 App 要支持多少个 CPU 架构)。如果很介意的话,可以考虑自行裁剪 OpenCV,然后再进行编译。

我所在的部门隶属于中台部门,主要输出接口和 SDK。在 SDK 中使用 OpenCV 的确会给业务方造成困扰,未来也会考虑如何减少 SDK 的体积,以及把 SDK 做成模块化。


关注【Java与Android技术栈】

更多精彩内容请关注扫码


以上是关于OpenCV 在 Android 上的应用的主要内容,如果未能解决你的问题,请参考以下文章

onBackPressed 上的 Android 片段 NullPointerException

opencv for android(十六):opencv在android上的录制avi格式的视频

OpenCV - arm/android 手机上的 JPEG 压缩不会产生正确的图像

在 Android 上的 OpenCV 中逐帧处理视频

android 上 OpenCV 的优化(neon 指令,tegra 3 上的 GLSL)

Android 7 上的 VideoView 错误