如何在Linux上为JNI应用程序编译动态库?

我正在使用

这就是我所做的。

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

您好

/* 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选项生成的,不是吗?

我试着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)

好了,要解决 更新#1 问题,显然我必须使用g++insted of

gccload虽然仍然无法使用该方法。我似乎无法告诉它正确的道路。

回答:

可以通过loadLibrary使用有效名称加载本机库。例如,对于Linux家族的lib

.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

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!

以上是 如何在Linux上为JNI应用程序编译动态库? 的全部内容, 来源链接: utcz.com/qa/404941.html

回到顶部