如何在Linux上编译一个JNI应用程序的dynamic库?

我使用的是Ubuntu 10.10

这就是我所做的。

Hello.java

class Hello { public native void sayHello(); static { System.loadLibrary("hellolib"); } public static void main(String[] args){ Hello h = new Hello(); h.sayHello(); } } 

然后我运行了下面的命令:

 dierre@cox:~/Scrivania/provajni$ javac Hello.java dierre@cox:~/Scrivania/provajni$ javah -jni Hello 

我已经获得了Hello.classHello.h

Hello.h

 /* DO NOT EDIT THIS FILE - it is machine generated */ #include <jni.h> /* Header for class Hello */ #ifndef _Included_Hello #define _Included_Hello #ifdef __cplusplus extern "C" { #endif /* * Class: Hello * Method: sayHello * Signature: ()V */ JNIEXPORT void JNICALL Java_Hello_sayHello (JNIEnv *, jobject); #ifdef __cplusplus } #endif #endif 

然后我创build了Hello.cpp

 #include <jni.h> #include "Hello.h" #include <iostream> using namespace std; JNIEXPORT void JNICALL Java_Hello_sayHello (JNIEnv *env, jobject obj) { cout << "Hello World!" << endl; return; } 

现在,我认为我搞砸了的部分。 我受到本指南的 启发编译dynamic或共享对象库部分) :

 dierre@cox:~/Scrivania/provajni$ gcc -I"/usr/lib/jvm/java-6-sun/include" -I"/usr/lib/jvm/java-6-sun/include/linux" -o hellolib.so -shared -Wl,-soname,hello.so Hello.cpp -static -lc 

生成文件hellolib.so

但是,当我尝试用java Hello运行它java Hello我有这个错误:

 Exception in thread "main" java.lang.UnsatisfiedLinkError: no hellolib in java.library.path at java.lang.ClassLoader.loadLibrary(ClassLoader.java:1734) at java.lang.Runtime.loadLibrary0(Runtime.java:823) at java.lang.System.loadLibrary(System.java:1028) at Hello.<clinit>(Hello.java:4) Could not find the main class: Hello. Program will exit. 

我甚至试过这个:

  LD_LIBRARY_PATH=`pwd` export LD_LIBRARY_PATH 

没有结果。

我知道我正在做一些非常愚蠢的事情,但我无法弄清楚它是什么。 dynamiclib是用-shared选项生成的,不是吗?

更新#1

我试过static { System.load("/home/dierre/Scrivania/provajni/hellolib.so"); } static { System.load("/home/dierre/Scrivania/provajni/hellolib.so"); }看看是否工作,但现在:

 Exception in thread "main" java.lang.UnsatisfiedLinkError: /home/dierre/Scrivania/provajni/hello.so: /home/dierre/Scrivania/provajni/hello.so: undefined symbol: _ZSt4cout at java.lang.ClassLoader$NativeLibrary.load(Native Method) at java.lang.ClassLoader.loadLibrary0(ClassLoader.java:1803) at java.lang.ClassLoader.loadLibrary(ClassLoader.java:1699) at java.lang.Runtime.load0(Runtime.java:770) at java.lang.System.load(System.java:1003) at Hello.<clinit>(Hello.java:4) 

更新#2好的,为了解决更新#1的问题,我不得不使用g++ insted gcc ,显然。 尽pipe如此,仍然无法使用load方法。 我似乎无法告诉它正确的道路。

Solutions Collecting From Web of "如何在Linux上编译一个JNI应用程序的dynamic库?"

本地库可以通过loadLibrary加载一个有效的名字。 例如,对于linux系列的lib XXXX .so,您的hellolib.so应该重命名为libhello.so。 顺便说一下,我用jni开发java,我将分离实现和本地接口(.c或.cpp)。

 static { System.loadLibrary("hello"); // will load libhello.so } 

实现头文件(HelloImpl.h):

 #ifndef _HELLO_IMPL_H #define _HELLO_IMPL_H #ifdef __cplusplus extern "C" { #endif void sayHello (); #ifdef __cplusplus } #endif #endif 

HelloImpl.cpp:

 #include "HelloImpl.h" #include <iostream> using namespace std; void sayHello () { cout << "Hello World!" << endl; return; } 

Hello.c(我更喜欢在c中编译jni):

 #include <jni.h> #include "Hello.h" #include "HelloImpl.h" JNIEXPORT void JNICALL Java_Hello_sayHello (JNIEnv *env, jobject obj) { sayHello(); return; } 

最后,我们可以通过一些步骤来编译它们:

  1. 编译obj(生成HelloImpl.o)

g ++ -c -I“/ opt / java / include”-I“/ opt / java / include / linux”HelloImpl.cpp

  1. 用.o编译jni

/ opt / java / include / linux“-o libhello.so -shared -Wl,-soname,hello.so Hello.c HelloImpl.o -static -lc

在步骤2中,我们使用g ++来编译它。 这个非常重要。 你可以看到如何混合使用C和C ++

编译完成后,可以用nm命令检查命名的功能:

 $ nm libhello.so |grep say 00000708 T Java_Hello_sayHello 00000784 t _GLOBAL__I_sayHello 00000718 T sayHello 

有一个标记为T的Java_Hello_sayHello,它应该与您的本地方法名称相同。 如果一切正常。 你可以运行它:

 $ java -Djava.library.path=. Hello Hello World! 

最后我的代码工作。 这是hello.java

 public class hello { public native void sayHello(int length) ; public static void main (String args[]) { String str = "I am a good boy" ; hello h = new hello () ; h.sayHello (str.length() ) ; } static { System.loadLibrary ( "hello" ) ; } } 

你应该把它编译为:

 $ javac hello.java 

要创建.h文件,你应该运行这个命令:

 $ javah -jni hello 

这是hello.h

 JNIEXPORT void JNICALL Java_hello_sayHello (JNIEnv *, jobject, jint); 

这里是hello.c

 #include<stdio.h> #include<jni.h> #include "hello.h" JNIEXPORT void JNICALL Java_hello_sayHello (JNIEnv *env, jobject object, jint len) { printf ( "\nLength is %d", len ); } 

要编译这个并创建一个共享库,我们必须运行这个命令:

 $ gcc -I/usr/lib/jvm/java-6-openjdk/include -o libhello.so -shared hello.c 

然后终于运行这个:

 $ java -Djava.library.path=. hello 

这抱怨C ++符号不可用。 我似乎记得,当我使用JNI的东西,所有的时间,有问题链接在C ++库,我们总是坚持普通的老C

如果你改变你的代码,使它是标准的C(并重命名文件):

 #include <jni.h> #include "Hello.h" #include <stdio.h> JNIEXPORT void JNICALL Java_Hello_sayHello (JNIEnv *env, jobject obj) { printf("Hello World"); return; } 

并编译它

 gcc -I/usr/lib/jvm/java-6-openjdk/include -o libhellolib.so -shared Hello.c 

有用

 java -Djava.library.path=`pwd` Hello Hello World