我正在使用 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.class和Hello.h。
Hello.class
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
然后我创建了 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; }
现在我认为我搞砸了。我从本指南(“ 编译动态或共享库” 部分)中得到了 启发 : __
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
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
没有结果。
我知道我在做一些非常愚蠢的事情,但我不知道这是什么。动态库是使用-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 of gcc。load虽然仍然无法使用该方法。我似乎无法告诉它正确的道路。
g++
gcc
load
可以通过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; }
最后,我们可以按以下步骤进行编译:
g ++ -c -I“ / opt / java / include” -I“ / opt / java / include / linux” HelloImpl.cpp
g ++ -I“ / opt / java / include” -I“ / 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!