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

我使用的是 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();
        }
}

然后我运行了 follwing 命令:

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

更新 #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++gccload


答案 1

本机库可以通过加载具有有效名称的库来加载。例如,libXXXX.因此对于linux系列,您的 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

您好.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. compile obj (generate 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!

答案 2

最后,我的代码可以正常工作。大家好.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 ( "\nLength is %d", len ); }

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

$ gcc -I/usr/lib/jvm/java-6-openjdk/include -o libhello.so -shared hello.c

然后最后运行这个:

$ java -Djava.library.path=. hello

推荐