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

How to resolve UnsatisfiedLinkError when using JNI with packages?

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

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

Sample.javamypackage 中看起来像这样:

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 header。然后我使用以下命令编译库:

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/ 的完整路径,但我得到了同样的错误。

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

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.5 上 运行 使用 clang 7.0.2java 1.8.0_101

您的库文件 (libSample.so) 中的名称似乎有误。

如果您使用:

System.loadLibrary("Sample");

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

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

System.mapLibraryName("Sample");

在目标平台上调用。

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