将JNI与包一起使用时如何解决UnsatisfiedLinkError?

时间:2017-06-08 15:40:33

标签: java c java-native-interface unsatisfiedlinkerror

首先,我的示例具有以下目录结构:

Sample.c
lib/
mypackage/
--Sample.java
Sample.java中的

mypackage如下所示:

package mypackage;

public class Sample {
    public static native int sampleMethod(int x);

    public static void main(String[] args) {
        System.loadLibrary("Sample");
        System.out.println("sampleMethod result: " + sampleMethod(5));
    }
}

我运行javac mypackage/Sample.java来编译java文件,javah mypackage.Sample生成JNI标头。然后我使用以下命令编译库:

clang -I"${JAVA_HOME}/include" -I"${JAVA_HOME}/include/darwin" -o lib/libSample.so -shared Sample.c

此时目录结构如下所示:

Sample.c
mypackage_Sample.h
lib/
--libSample.so
mypackage/
--Sample.java
--Sample.class

现在,当我尝试使用java -Djava.library.path=./lib/ mypackage.Sample运行示例时,出现以下错误:

Exception in thread "main" java.lang.UnsatisfiedLinkError: no Sample in java.library.path
    at java.lang.ClassLoader.loadLibrary(ClassLoader.java:1867)
    at java.lang.Runtime.loadLibrary0(Runtime.java:870)
    at java.lang.System.loadLibrary(System.java:1122)
    at mypackage.Sample.main(Sample.java:7)

我尝试指定lib/的完整路径,但我得到了同样的错误。

我不确定标题和实现的代码是否重要,但无论如何我都会发布它们。

mypackage_Sample.h

/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class mypackage_Sample */

#ifndef _Included_mypackage_Sample
#define _Included_mypackage_Sample
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     mypackage_Sample
 * Method:    sampleMethod
 * Signature: (I)I
 */
JNIEXPORT jint JNICALL Java_mypackage_Sample_sampleMethod
  (JNIEnv *, jclass, jint);

#ifdef __cplusplus
}
#endif
#endif

Sample.c

#include "mypackage_Sample.h"
#include <stdio.h>

JNIEXPORT jint JNICALL Java_mypackage_Sample_sampleMethod
  (JNIEnv * env, jclass obj, jint num) {
      return num * num;
  }

我使用OS X Yosemite 10.10.5clang 7.0.2java 1.8.0_101上运行此功能。

1 个答案:

答案 0 :(得分:1)

您的库文件(libSample.so)看起来有错误的名称。

如果您使用:

System.loadLibrary("Sample");

JVM会将此名称映射到特定于平台的文件名以尝试加载。在libSample.so的Linux上,在Sample.dll的Windows上,但在OS X上,它是其他内容。

您可以通过查看以下输出找出您的库文件应具有的名称:

System.mapLibraryName("Sample");

在目标平台上调用。

之后,您可以将其用作库文件的名称。