我想创建一个Android应用程序,它使用原生(使用C ++)BOW + SVM进行预测。不幸的是我在构建原生部分时遇到了问题。由于OpenCV SDK for Android中不包含非免费模块,因此我需要使用this tutorial自行构建模块。好像我成功构建了.so文件。这是输出:
[armeabi-v7a] Prebuilt : libopencv_java.so <= /home/crash-id/Development/SDK/OpenCV-2.4.9-android-sdk/sdk/native/jni/../libs/armeabi-v7a/
[armeabi-v7a] SharedLibrary : libnonfree.so
[armeabi-v7a] Install : libnonfree.so => libs/armeabi-v7a/libnonfree.so
[armeabi-v7a] Install : libopencv_java.so => libs/armeabi-v7a/libopencv_java.so
所以问题来了,我必须将这个.so文件添加到我的项目中。我将libnonfree.so添加到了我的jni文件夹中。然后我编辑了Android.mk。在这里,我提供了我的.mk文件。
Android.mk
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE := nonfree_prebuilt
LOCAL_SRC_FILES := libnonfree.so
include $(PREBUILT_SHARED_LIBRARY)
include $(CLEAR_VARS)
OPENCV_CAMERA_MODULES := on
OPENCV_INSTALL_MODULES := on
include /home/crash-id/Development/SDK/OpenCV-2.4.9-android-sdk/sdk/native/jni/OpenCV.mk
#LOCAL_SHARED_LIBRARIES := nonfree_prebuilt #if I add this, it says undefined reference for everything in the cv namespace.
LOCAL_SRC_FILES := SVMDetector.cpp
LOCAL_MODULE := svm_detector
LOCAL_C_INCLUDES += /home/crash-id/Development/SDK/OpenCV-2.4.9-android-sdk/sdk/native/jni/include
LOCAL_CFLAGS := -Werror -O3 -ffast-math
LOCAL_LDLIBS += -llog -ldl
include $(BUILD_SHARED_LIBRARY)
Application.mk
APP_STL := gnustl_static
APP_CPPFLAGS := -frtti -fexceptions
APP_ABI := armeabi-v7a
APP_PLATFORM := android-15
但这不起作用。当我尝试构建应用程序时,出现以下错误:
./obj/local/armeabi-v7a/objs/svm_detector/SVMDetector.o: in function Java_org_elsys_thesisdiploma_cammect_FrameProcess_SVMDetect:jni/SVMDetector.cpp:23: error: undefined reference to 'cv::initModule_nonfree()'
当我点击initModule_nonfree();
上的右键时,Eclipse会打开nonfree.hpp文件,这是它的内容:
#ifndef __OPENCV_NONFREE_HPP__
#define __OPENCV_NONFREE_HPP__
#include "opencv2/nonfree/features2d.hpp"
namespace cv
{
CV_EXPORTS_W bool initModule_nonfree();
}
#endif
但是我不确定链接器知道这个方法的实现在哪里。由于它给出了错误,因此它没有。
修改
如果我添加LOCAL_ALLOW_UNDEFINED_SYMBOLS := true
,项目会成功编译,但会导致运行时错误:
02-17 00:15:58.197: E/AndroidRuntime(8793): FATAL EXCEPTION: main
02-17 00:15:58.197: E/AndroidRuntime(8793): Process: com.example.cammect, PID: 8793
02-17 00:15:58.197: E/AndroidRuntime(8793): java.lang.UnsatisfiedLinkError: dlopen failed: cannot locate symbol "_ZN2cv18initModule_nonfreeEv" referenced by "libsvm_detector.so"...
你知道我做错了什么吗?提前谢谢!
答案 0 :(得分:15)
D:\adt-bundle-windows-x86_64-20140702\android-ndk-r10d\
)D:\CODE\OpenCV-2.4.10-android-sdk\
),Download link D:\CODE\OpenCV-2.4.10\
),Download link 我们实际上只需要将OpenCV-2.4.10
源代码中的一些文件复制到OpenCV-2.4.10-android-sdk
,即:
将nonfree文件夹从OpenCV-2.4.10\sources\modules\nonfree\include\opencv2\
复制到OpenCV-2.4.10-android-sdk\sdk\native\jni\include\opencv2
。
创建一个文件夹来保存libnonfree.so
的新项目。在这里,我称之为libnonfree
。在jni
下创建一个libnonfree
文件夹。将以下文件从OpenCV-2.4.10\sources\modules\nonfree\src
复制到libnonfree\jni\
文件夹:
建立libnonfree.so
:
创建Android.mk
和Application.mk
脚本。此Android.mk
用于构建libnonfree.so
。
OPENCV_PATH
所在的OpenCV-2.4.10-android-sdk
cd
进入项目文件夹libnonfree
并输入ndk-build
以构建libnonfree.so
。
到目前为止,您已在libnonfree.so
文件夹中获得libopencv_java.so
以及libgnustl_shared.so
和libnonfree\libs\armeabi-v7a
。
您可以使用这些库轻松构建任何SIFT或SURF应用程序。如果你想在你的Android应用程序中使用JAVA代码中的SIFT和SURF,你只需要为你想要使用的函数编写JNI接口。
创建项目文件夹调用libnonfree_demo
。在项目文件夹中创建一个jni
文件夹。然后将libnonfree.so
以及libopencv_java.so
和libgnustl_shared.so
复制到jni
。
在jni
中创建nonfree_jni.cpp。这是简单的SIFT测试程序。它基本上读取图像并检测关键点,然后提取特征描述符,最后将关键点绘制到输出图像。
在Android.mk
内创建Application.mk
和jni
:
OPENCV_PATH
所在的OpenCV-2.4.10-android-sdk
cd
进入项目文件夹libnonfree_demo
并输入ndk-build
以构建libnonfree_demo.so
。
此时,您可以使用SVMDetector
轻松扩展示例应用。只需将源文件和包含文件int复制到文件夹libnonfree_demo\jni
,然后将cpp文件添加到LOCAL_SRC_FILES
中的Android.mk
。
整个来源可以从以下网址下载:https://github.com/bkornel/opencv_android_nonfree。
来自:http://web.guohuiwang.com/technical-notes/sift_surf_opencv_android
的原始来源答案 1 :(得分:3)
我可以补充一点,为了在正在运行的应用程序中使用新库,需要执行以下步骤:
1)在您的文件夹libnonfree / libs / [TARGET PLATFORM] /中,现在有3个文件: - libgnustl_shared.so - libnonfree.so - libopencv_java.so
在你自己的项目中(我的IDE是Android Studio),你有一个文件夹src / main /,带有子文件夹: - java - res
创建一个新文件夹(如果还没有):“jniLibs”[此文件夹由Gradle自动解析]
将“libnonfree / libs /”下的3个上述文件夹复制到“jniLibs”文件夹中。你最终得到了这样的结构: screenshot of the jniLibs folder
/ app / src / main / jniLibs / [armeabi,armeabi-v7a,...] / [libgnustl_shared.so,libopencv_java.so,libnonfree.so]
2)在代码的某处,你有一条这样的一行:
OpenCVLoader.initAsync(OpenCVLoader.OPENCV_VERSION_2_4_10, this, mLoaderCallback);
此行告诉您的应用从本地安装的OpenCV Manager动态加载预编译库。为了使用自编译的非自由版本,我们用以下内容替换上面的行:
if(!OpenCVLoader.initDebug())
{
}
else
{
System.loadLibrary("nonfree");
}
现在,我们确保使用我们在应用程序中提供的非自由库。
3)好吧,运行一个SURF描述符:
Bitmap mPhotograph = BitmapFactory.decodeFile(_image_path);
Mat real_image = new Mat();
Utils.bitmapToMat(mPhotograph, real_image);
MatOfKeyPoint keypoints_real = new MatOfKeyPoint();
FeatureDetector detector = FeatureDetector.create(FeatureDetector.SURF);
detector.detect(real_image, keypoints_real);
之前,应用程序会返回一个错误的信号,这次它完成了它的工作,你可以评估产生的关键点。