Как скомпилировать динамическую библиотеку для приложения JNI на Linux? - PullRequest
37 голосов
/ 17 октября 2010

Я использую 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.

Hello.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++ вместо gcc, очевидно.По-прежнему возникают проблемы с использованием метода load.Я не могу сказать, что это правильный путь.

Ответы [ 3 ]

40 голосов
/ 21 октября 2010

Собственная библиотека может быть загружена с помощью loadLibrary с допустимым именем. Например, lib XXXX .so для семейства Linux, ваш hellolib.so следует переименовать в libhello.so. Кстати, я разрабатываю Java с jni, я отделю реализацию и нативный интерфейс (.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 (я предпочитаю компилировать jni в c):

#include <jni.h>
#include "Hello.h"
#include "HelloImpl.h"

JNIEXPORT void JNICALL Java_Hello_sayHello (JNIEnv *env, jobject obj) {
    sayHello();
    return;
}

Наконец, мы можем скомпилировать их в несколько шагов:

  1. compile obj (генерировать HelloImpl.o)

g ++ -c -I "/ opt / java / include" -I "/ opt / java / include / linux" HelloImpl.cpp

  1. скомпилировать JNI с .o

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!
27 голосов
/ 11 декабря 2011

Наконец-то мой код работает.Это 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

is 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
5 голосов
/ 17 октября 2010

Это жалуется на недоступность символов 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
...