Jak skompilować bibliotekę dynamiczną dla aplikacji JNI na Linuksie?

Używam Ubuntu 10.10

Więc to właśnie zrobiłem.

Witam.java :

class Hello {
        public native void sayHello();

        static { System.loadLibrary("hellolib"); }

        public static void main(String[] args){
                Hello h = new Hello();
                h.sayHello();
        }
}

Potem uruchomiłem komendę follwing:

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

dierre@cox:~/Scrivania/provajni$ javah -jni Hello 

Uzyskałem Hello.class i Hello.h.

Witam.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

Potem stworzyłem Witam.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;
}

A teraz część, w której myślę, że nawaliłem. Zainspirowałem się tym przewodnikiem (skompilowałem dynamiczną lub współdzieloną bibliotekę obiektów sekcja):

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

Który generuje plik hellolib.so

Ale kiedy próbuję uruchomić go z java Hello mam taki błąd:

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.

Próbowałem nawet tego:

  LD_LIBRARY_PATH=`pwd`
  export LD_LIBRARY_PATH

Bez wyników.

Wiem, że robię coś bardzo głupiego, ale nie wiem, co to jest. Dynamiczna lib jest generowana z opcją-shared, prawda?

Aktualizacja #1

Próbowałem static { System.load("/home/dierre/Scrivania/provajni/hellolib.so"); } zobaczyć, czy to działa, ale teraz:

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)

Aktualizacja #2 Ok, aby rozwiązać problem Update #1 musiałem oczywiście użyć g++ input of gcc. Nadal jednak mam problemy z użyciem metody load. Nie potrafię wskazać właściwej drogi.

Author: dierre, 2010-10-17

3 answers

Natywna biblioteka może być załadowana przez loadLibrary z poprawną nazwą. Na przykład lib XXXX. więc dla rodziny Linuksa, Twój hellolib.so należy zmienić nazwę na libhello.so. Nawiasem mówiąc, rozwijam Javę z jni, oddzielę implementację i natywny interfejs (.c lub .cpp).

static {
    System.loadLibrary("hello"); // will load libhello.so
}

Nagłówek implementacji (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;
}
Witam.c (wolę kompilować jni w c):
#include <jni.h>
#include "Hello.h"
#include "HelloImpl.h"

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

Wreszcie możemy skompilować je w jakimś kroki:

  1. compile obj (generate HelloImpl.o)

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

  1. skompiluj jni z .o

G++-I " / opt / java / include" -I "/opt / java / include / linux " - o libhello.so -wspólne - WL, - soname,hello.so Witam.C HelloImpl.o-static-LC

W Kroku 2 używamy g++ do kompilacji. To bardzo ważne. yor można zobaczyć Jak wymieszać C i C++

Po kompilacji można sprawdzić nazwę funkcji za pomocą nm:

$ nm libhello.so |grep say
00000708 T Java_Hello_sayHello
00000784 t _GLOBAL__I_sayHello
00000718 T sayHello

Istnieje Java_Hello_sayHello oznaczony T. powinien być extactly równy Twojej natywnej nazwie metody. Jeśli wszystko jest w porządku. można go uruchomić:

$ java -Djava.library.path=. Hello
Hello World!
 38
Author: qrtt1,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2010-10-21 12:44:05

W końcu mój kod działa. Tu 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" ) ;
  }
}

Należy skompilować jako:

$ javac hello.java 

Do tworzenia .plik h powinieneś uruchomić to polecenie:

$ javah -jni hello

To jest hello.h:

JNIEXPORT void JNICALL Java_hello_sayHello
(JNIEnv *, jobject, jint);

Oto 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 ); }

Aby to skompilować i stworzyć bibliotekę współdzieloną musimy wykonać następujące polecenie:

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

W końcu uruchom ten:

$ java -Djava.library.path=. hello
 26
Author: Sandipan,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2013-08-14 14:50:56

To narzeka, że symbole C++ nie są dostępne. Wydaje mi się, że kiedy używam JNI rzeczy przez cały czas, że były problemy z linkowaniem w bibliotekach C++ i zawsze trzymaliśmy się zwykłego starego C

Jeśli zmienisz kod tak, aby był to standard C (i zmienisz nazwę pliku):

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

JNIEXPORT void JNICALL Java_Hello_sayHello (JNIEnv *env, jobject obj) {
        printf("Hello World");
        return;
}

I skompilować

gcc -I/usr/lib/jvm/java-6-openjdk/include  -o libhellolib.so -shared Hello.c

It works

java -Djava.library.path=`pwd` Hello
Hello World
 4
Author: Gordon Thompson,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2010-10-18 12:13:33