使用OpenCV 2.4.2的Android UnsatisfiedLinkError

时间:2012-07-23 14:09:36

标签: android opencv unsatisfiedlinkerror

只是想尝试一个简单的openCV android程序。在instructions here之后下载并安装了OpenCV for Android,并将OpenCV Library 2.4.2添加为我自己的android项目的库项目,如指令所述。

但是当我编译标准的“Hello World Program”时,如下所示, 如果我包含Mat mat = new Mat();行,则会失败,否则会成功。

package com.example;

import org.opencv.core.Mat;

import android.app.Activity;
import android.os.Bundle;

public class HelloAndroidActivity extends Activity {
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);

        Mat mat = new Mat();
    }
}

这是打印出的堆栈跟踪:

    07-23 09:59:43.835: E/AndroidRuntime(8222): FATAL EXCEPTION: main
07-23 09:59:43.835: E/AndroidRuntime(8222): java.lang.UnsatisfiedLinkError: n_Mat
07-23 09:59:43.835: E/AndroidRuntime(8222):     at org.opencv.core.Mat.n_Mat(Native Method)
07-23 09:59:43.835: E/AndroidRuntime(8222):     at org.opencv.core.Mat.<init>(Mat.java:181)
07-23 09:59:43.835: E/AndroidRuntime(8222):     at com.example.HelloAndroidActivity.onCreate(HelloAndroidActivity.java:15)
07-23 09:59:43.835: E/AndroidRuntime(8222):     at android.app.Activity.performCreate(Activity.java:4538)
07-23 09:59:43.835: E/AndroidRuntime(8222):     at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1071)
07-23 09:59:43.835: E/AndroidRuntime(8222):     at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2161)
07-23 09:59:43.835: E/AndroidRuntime(8222):     at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2240)
07-23 09:59:43.835: E/AndroidRuntime(8222):     at android.app.ActivityThread.access$600(ActivityThread.java:139)
07-23 09:59:43.835: E/AndroidRuntime(8222):     at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1262)
07-23 09:59:43.835: E/AndroidRuntime(8222):     at android.os.Handler.dispatchMessage(Handler.java:99)
07-23 09:59:43.835: E/AndroidRuntime(8222):     at android.os.Looper.loop(Looper.java:154)
07-23 09:59:43.835: E/AndroidRuntime(8222):     at android.app.ActivityThread.main(ActivityThread.java:4977)
07-23 09:59:43.835: E/AndroidRuntime(8222):     at java.lang.reflect.Method.invokeNative(Native Method)
07-23 09:59:43.835: E/AndroidRuntime(8222):     at java.lang.reflect.Method.invoke(Method.java:511)
07-23 09:59:43.835: E/AndroidRuntime(8222):     at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:784)
07-23 09:59:43.835: E/AndroidRuntime(8222):     at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:551)
07-23 09:59:43.835: E/AndroidRuntime(8222):     at dalvik.system.NativeStart.main(Native Method)

有两点需要注意:我没有直接使用此代码中的任何原生代码(就像这里的其他一些问题一样),旧的OpenCV 2.3.x库在使用相同的方法之前工作得很好。两个Android项目都具有相同的目标和支持的API设置。

3 个答案:

答案 0 :(得分:19)

想出来。没有静态链接库。如果您使用此代码,则可以使用。

package com.example;

import org.opencv.android.BaseLoaderCallback;
import org.opencv.android.LoaderCallbackInterface;
import org.opencv.android.OpenCVLoader;

import android.app.Activity;
import android.os.Bundle;
import android.util.Log;

public class HelloAndroidActivity extends Activity
{

    final String TAG = "Hello World";

private BaseLoaderCallback mOpenCVCallBack = new BaseLoaderCallback(this) {
@Override
public void onManagerConnected(int status) {
   switch (status) {
       case LoaderCallbackInterface.SUCCESS:
       {
      Log.i(TAG, "OpenCV loaded successfully");
      // Create and set View
      setContentView(R.layout.main);
       } break;
       default:
       {
      super.onManagerConnected(status);
       } break;
   }
    }
};

/** Called when the activity is first created. */
@Override
public void onCreate(Bundle savedInstanceState)
{
    Log.i(TAG, "onCreate");
    super.onCreate(savedInstanceState);

    Log.i(TAG, "Trying to load OpenCV library");
    if (!OpenCVLoader.initAsync(OpenCVLoader.OPENCV_VERSION_2_4_2, this, mOpenCVCallBack))
    {
      Log.e(TAG, "Cannot connect to OpenCV Manager");
    }
}
}

但是,不太喜欢这种“OpenCV Manager”的想法。这使得用户必须在应用程序运行之前手动安装多个软件包。

答案 1 :(得分:3)

解决方案是在@Jason回答中使用OpenCV Manager。关于official documentation here的详细解释也是如此。

但是就像@Jason所说的那样,“这使得用户必须在应用程序运行之前手动安装多个软件包”。尽管如此, OpenCV Manager 具有一些优势,例如:

  • 如果更新了OpenCV,用户只需要更新管理器/库。使用管理器的应用程序可以保持不变。

  • 您的应用apk尺寸会小很多:

    • 对于为您的设备架构编译的OpenCV库,一个简单的opencv应用程序对于每个应用程序大约约400KB + ~800KB用于OpenCV Manager + ~12MB。
    • 使用传统的静态链接设备中的每个 opencv应用至少约为25MB。
    • 当然,这些尺寸取决于您放置在应用中的内容数量......

即便如此,如果您希望以传统方式部署应用,静态链接you can read the instructions here

答案 2 :(得分:2)

 static{System.loadLibrary("opencv_java3"); } //the name of the .so file, without the 'lib' prefix

您可以在活动的任何位置静态加载打开的cv库。 搜索jniLibs文件夹中的.so文件,并将其复制/粘贴为“loadLibrary”方法的参数,不带lib前缀。