首先,我的示例具有以下目录结构:
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.5
和clang 7.0.2
在java 1.8.0_101
上运行此功能。
答案 0 :(得分:1)
您的库文件(libSample.so
)看起来有错误的名称。
如果您使用:
System.loadLibrary("Sample");
JVM会将此名称映射到特定于平台的文件名以尝试加载。在libSample.so
的Linux上,在Sample.dll
的Windows上,但在OS X上,它是其他内容。
您可以通过查看以下输出找出您的库文件应具有的名称:
System.mapLibraryName("Sample");
在目标平台上调用。
之后,您可以将其用作库文件的名称。