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

die*_*rre 38 c++ java linux java-native-interface compilation

我正在使用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();
        }
}
Run Code Online (Sandbox Code Playgroud)

然后我运行了以下命令:

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

dierre@cox:~/Scrivania/provajni$ javah -jni Hello 
Run Code Online (Sandbox Code Playgroud)

我获得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
Run Code Online (Sandbox Code Playgroud)

然后我创建了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;
}
Run Code Online (Sandbox Code Playgroud)

而现在,我认为我搞砸了.我受到指南的启发(编译动态或共享对象库部分):

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
Run Code Online (Sandbox Code Playgroud)

生成文件 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.
Run Code Online (Sandbox Code Playgroud)

我甚至试过这个:

  LD_LIBRARY_PATH=`pwd`
  export LD_LIBRARY_PATH
Run Code Online (Sandbox Code Playgroud)

没有结果.

我知道我做的事非常愚蠢,但我无法弄清楚它是什么.使用-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)
Run Code Online (Sandbox Code Playgroud)

更新#2 好了,显然,为了解决我必须使用insted 的更新#1问题.但是仍然有使用该方法的麻烦.我似乎无法告诉它正确的道路.g++gccload

qrt*_*tt1 40

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

static {
    System.loadLibrary("hello"); // will load libhello.so
}
Run Code Online (Sandbox Code Playgroud)

实现头(HelloImpl.h):

#ifndef _HELLO_IMPL_H
#define _HELLO_IMPL_H

#ifdef __cplusplus
        extern "C" {
#endif

        void sayHello ();

#ifdef __cplusplus
        }
#endif

#endif
Run Code Online (Sandbox Code Playgroud)

HelloImpl.cpp:

#include "HelloImpl.h"
#include  <iostream>

using namespace std;

void sayHello () {
    cout << "Hello World!" << endl;
    return;
}
Run Code Online (Sandbox Code Playgroud)

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;
}
Run Code Online (Sandbox Code Playgroud)

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

  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
Run Code Online (Sandbox Code Playgroud)

有一个标记为T的Java_Hello_sayHello.它应该与您的本机方法名称相等.如果一切都好.你可以运行它:

$ java -Djava.library.path=. Hello
Hello World!
Run Code Online (Sandbox Code Playgroud)

  • 如果我在上一个编译步骤中省略了"-static"选项,这对我有用.如果我不这样做,我会收到此错误:/ usr/bin/ld:/usr/lib/gcc/x86_64-linux-gnu/4.6.1/crtbeginT.o:对于`__DTOR_END__'重新定位R_X86_64_32时无法使用制作共享对象; 用-fPIC重新编译我在这个SO问题中找到了解决方案:http://stackoverflow.com/questions/6634387/c-statically-linked-shared-library (4认同)

小智 29

最后我的代码工作.这是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" ) ;
  }
}
Run Code Online (Sandbox Code Playgroud)

您应该将其编译为:

$ javac hello.java 
Run Code Online (Sandbox Code Playgroud)

要创建.h文件,您应该运行以下命令:

$ javah -jni hello
Run Code Online (Sandbox Code Playgroud)

这是hello.h:

JNIEXPORT void JNICALL Java_hello_sayHello
(JNIEnv *, jobject, jint);
Run Code Online (Sandbox Code Playgroud)

这是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 ); }
Run Code Online (Sandbox Code Playgroud)

要编译它并创建共享库,我们必须运行以下命令:

$ gcc -I/usr/lib/jvm/java-6-openjdk/include -o libhello.so -shared hello.c
Run Code Online (Sandbox Code Playgroud)

然后最后运行这个:

$ java -Djava.library.path=. hello
Run Code Online (Sandbox Code Playgroud)


Gor*_*son 5

这抱怨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;
}
Run Code Online (Sandbox Code Playgroud)

并编译它

gcc -I/usr/lib/jvm/java-6-openjdk/include  -o libhellolib.so -shared Hello.c
Run Code Online (Sandbox Code Playgroud)

有用

java -Djava.library.path=`pwd` Hello
Hello World
Run Code Online (Sandbox Code Playgroud)

  • 它是gcc而不是g ++。但是我仍然不能使用`loadLibrary`,只能使用`load` (2认同)