我正在使用 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
。
你好.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;
}
现在我认为我搞砸了。我被这个guide (Compile the Dynamic or Shared Object Library section) 启发 :
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
运行它时,我遇到了这个错误:
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"); }
看看是否可行,但现在:
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
好的,为了解决 Update #1 问题,很明显,我必须使用 gcc
的 g++
insted。但是仍然无法使用 load
方法。我似乎无法告诉它正确的路径。
最佳答案
可以通过 loadLibrary 使用有效名称加载 native 库。例如,对于 linux 系列,libXXXX.so,你的 hellolib.so 应该重命名为 libhello.so。 顺便说一句,我用jni开发java,我会把实现和native interface(.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;
}
最后,我们可以分几步编译它们:
- 编译obj(生成HelloImpl.o)
g++ -c -I"/opt/java/include" -I"/opt/java/include/linux" HelloImpl.cpp
- 用 .o 编译 jni
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++ 对其进行编译。这个非常重要。你可以看到How to mix C and C++
编译后可以用nm查看函数命名:
$ nm libhello.so |grep say
00000708 T Java_Hello_sayHello
00000784 t _GLOBAL__I_sayHello
00000718 T sayHello
有一个标记为 T 的 Java_Hello_sayHello。它应该完全等于您的 native 方法名称。如果一切正常。你可以运行它:
$ java -Djava.library.path=. Hello
Hello World!
关于java - 如何在 linux 上为 JNI 应用程序编译动态库?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3950635/