如何在Linux上为JNI应用程序编译动态库?
- 2024-11-01 08:41:00
- admin 原创
- 39
问题描述:
我正在使用Ubuntu 10.10
所以我就是这么做的。
你好.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;
}
现在到了我认为我搞砸了的部分。我受到了本指南的启发(编译动态或共享对象库部分):
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
好的,为了解决更新 #1 的问题,我显然必须使用g++
ins 而gcc
不是 。但是使用该方法仍然有问题load
。我似乎无法告诉它正确的路径。
解决方案 1:
本机库可以通过 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;
}
最后,我们可以按照以下几个步骤来编译它们:
编译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++ 来编译它。这非常重要。你可以查看如何混合 C 和 C++
编译后,你可以用nm检查函数命名:
$ nm libhello.so |grep say
00000708 T Java_Hello_sayHello
00000784 t _GLOBAL__I_sayHello
00000718 T sayHello
有一个 Java_Hello_sayHello 标记为 T。它应该完全等于您的本机方法名称。如果一切正常。您可以运行它:
$ java -Djava.library.path=. Hello
Hello World!
解决方案 2:
我的代码终于可以运行了。这是 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 ( "
Length is %d", len ); }
要编译它并创建共享库,我们必须运行以下命令:
$ gcc -I/usr/lib/jvm/java-6-openjdk/include -o libhello.so -shared hello.c
然后最后运行这个:
$ java -Djava.library.path=. hello
解决方案 3:
这抱怨 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
- 2024年20款好用的项目管理软件推荐,项目管理提效的20个工具和技巧
- 2024年开源项目管理软件有哪些?推荐5款好用的项目管理工具
- 项目管理软件有哪些?推荐7款超好用的项目管理工具
- 项目管理软件哪个最好用?盘点推荐5款好用的项目管理工具
- 项目管理软件有哪些最好用?推荐6款好用的项目管理工具
- 项目管理软件有哪些,盘点推荐国内外超好用的7款项目管理工具
- 2024项目管理软件排行榜(10类常用的项目管理工具全推荐)
- 项目管理软件排行榜:2024年项目经理必备5款开源项目管理软件汇总
- 2024年常用的项目管理软件有哪些?推荐这10款国内外好用的项目管理工具
- 项目管理必备:盘点2024年13款好用的项目管理软件