Как скомпилировать динамическую библиотеку для приложения JNI в Linux?
Я использую 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();
}
}
Затем я выполнил команды follwing:
[email protected]:~/Scrivania/provajni$ javac Hello.java
[email protected]:~/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;
}
И теперь часть, где, я думаю, я прищурился. Я был вдохновлен этим guide (Компилировать раздел динамической или общей библиотеки объектов):
[email protected]:~/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
Хорошо, чтобы решить проблему Update # 1, я должен был использовать g++
insted из gcc
, очевидно. Тем не менее, есть проблема с использованием метода load
. Я не могу сказать, что это правильный путь.
Ответы
Ответ 1
Нативная библиотека может быть загружена 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;
}
Наконец, мы можем скомпилировать их в несколько шагов:
- compile obj (сгенерировать HelloImpl.o)
g++ -c -I "/opt/java/include" -I "/opt/java/include/linux" HelloImpl.cpp
- скомпилировать 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++ для его компиляции. Это очень важно. yor может видеть Как смешивать 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 ( "\nLength is %d", len ); }
Чтобы скомпилировать это и создать общую библиотеку, мы должны запустить эту команду:
$ gcc -I/usr/lib/jvm/java-6-openjdk/include -o libhello.so -shared hello.c
Затем, наконец, запустите это:
$ java -Djava.library.path=. hello
Ответ 3
Это говорит о том, что символы С++ недоступны. Я, кажется, помню, когда я все время делал JNI, когда возникали проблемы с связыванием в С++-библиотеках, и мы всегда придерживались простого старого 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