PDA

View Full Version : [java] load dll library


topix93
16-05-2012, 15:45
String path = Inizio.class.getProtectionDomain().getCodeSource().getLocation().getPath();
path = path.substring(0, path.lastIndexOf('/'));
path = path.substring(1, path.lastIndexOf('/'));
System.out.println(path);
System.loadLibrary(path + "/lib/jnotify.dll");
System.loadLibrary(path + "/lib/jnotify.dll");

Ricevo il seguente errore

Exception in thread "main" java.lang.UnsatisfiedLinkError: Exception in thread "main" java.lang.UnsatisfiedLinkError: no C:/JAVA/DirectoryListener/lib/jnotify_64bit in java.library.path
at java.lang.ClassLoader.loadLibrary(Unknown Source)
at java.lang.Runtime.loadLibrary0(Unknown Source)
at java.lang.System.loadLibrary(Unknown Source)

cosa ho sbagliato?

PGI-Bis
16-05-2012, 17:30
Prova con System.load al posto di loadLibrary. loadLibrary prende il nome che gli passi e lo trasforma affinchè coincida con il nome di una libreria dinamica valida per il sistema in uso (nel tuo caso noti dalla traccia dell'eccezione che l'ha fatto diventare il nome di una dll per un sistema windows a 64 bit, su un unix a 32 bit l'avrebbe fatto diventare un .so).

topix93
17-05-2012, 08:04
Il fatto e che io sto utilizzando la libreria jnotify che necessita di almeno una delle due dll citate in precedenza. Ho provato a fare come hai detto tu con System.load ma pur facendo in questo modo il mio programma va in errore...
Quindi ci possono essere alcune soluzioni. o metto la dll in alcune directory predefinite come system32 oppure le lascio all'interno dellla cartella principale del mio progetto. facendo in quest'ultimo modo però, quando esporto il mio progetto in un file jar, questo non funziona piu

Come posso fare quindi per risolvere questa situazione?

PGI-Bis
17-05-2012, 09:08
Uomo di poca fede! :)

/*
* To change this template, choose Tools | Templates
* and open the template in the editor.
*/
package test;

import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.nio.file.Files;
import java.util.logging.Level;
import java.util.logging.Logger;
import net.contentobjects.jnotify.JNotify;
import net.contentobjects.jnotify.JNotifyListener;

/**
*
* @author utente4
*/
public class Test {

/**
* @param args the command line arguments
*/
public static void main(String[] args) throws Exception {
String libraryName = System.mapLibraryName("jnotify");//occhio a linux 64 bit: la "dll" ha lo stesso nome di quella a 32
String source = "/natives/" + libraryName;
File out = new File(libraryName);
if(!out.exists()) try(InputStream in = Test.class.getResourceAsStream(source);) {
Files.copy(in, out.toPath());
} catch(IOException ex) {
Logger.getLogger(Test.class.getName()).log(Level.INFO, "Errore", ex);
}
System.load(out.getCanonicalPath());

String path = System.getProperty("user.home");
int id = JNotify.addWatch(path,JNotify.FILE_CREATED | JNotify.FILE_DELETED, false, new JNotifyListener() {

@Override
public void fileCreated(int i, String string, String string1) {
System.out.println("Bongo!");
}

@Override
public void fileDeleted(int i, String string, String string1) {
System.out.println("bingo!");
}

@Override
public void fileModified(int i, String string, String string1) {
}

@Override
public void fileRenamed(int i, String string, String string1, String string2) {
}
});
Thread.sleep(60000);
JNotify.removeWatch(id);
}
}

Questo funziona perfettamente. L'unico intoppo sta nel fatto che poichè in linux le librerie a 64 bit hanno lo stesso nome di quelle a 32 bisognerebbe aggiungere una subroutine per pigliare quella giusta (in base alla jvm) ma è un dettaglio.
Il codice presuppone che nel jar (o in uno dei jar del programma) ci sia una cartella "natives" con dentro le varie dll

natives/jnotify.dll
natives/jnotify_64bit.dll
natives/libjnotify.jnilib
natives/libnotify.so

(mancherebbe quella per linux 64bit). Il codice estrae la dll dal jar se manca e poi la carica.

topix93
17-05-2012, 20:38
Se non ci fossi te cosa farei?? :)

comunque non ho capito questa parte di codice
if(!out.exists()) try(InputStream in = Test.class.getResourceAsStream(source);) {
Files.copy(in, out.toPath());
} catch(IOException ex) {
Logger.getLogger(Test.class.getName()).log(Level.INFO, "Errore", ex);
}
Ho notato che Files.copy fa riferimento a java7, c'e un equivalente per java6??

PGI-Bis
17-05-2012, 20:51
Banalmente copia i byte del file zippato nel jar nel file corrispondente ma fuori dal jar. Non c'è l'equivalente in Java 6, puoi dire (vado a braccio):

InputStream in = getClass().getResource("/natives/" + nomeDellaDll);
File out = new File(nomeDellaDll);
FileOutputStream fout = null;
try {
fout = new FileOutputStream(out);
for(int b = in.read(); b != -1; b = in.read()) fout.write(b);
} catch(IOException ex) {
...
} finally {
if(out != null) try { out.close(); } catch(IOException ignore) {}
}

topix93
18-05-2012, 14:57
InputStream in = Test.class.getResourceAsStream(source);
mi restituisce nullpointerException su in... come mai??

questa e la mia classe
String libraryName = System.mapLibraryName("jnotify");//occhio a linux 64 bit: la "dll" ha lo stesso nome di quella a 32
String source = "/natives/" + libraryName;
File out = new File(libraryName);
if(!out.exists()) {
FileOutputStream fout = null;
try{
InputStream in = Test.class.getResourceAsStream(source);
fout = new FileOutputStream(out);
for(int b = in.read(); b != -1; b = in.read())
fout.write(b);
} catch(IOException ex) {
Logger.getLogger(Test.class.getName()).log(Level.INFO, "Errore", ex);
}
}
try {
System.out.println(out.getCanonicalPath());
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}

PGI-Bis
18-05-2012, 17:10
nel programma di esempio che ho testato si presuppone che il jar contenga una cartella "natives" (direttamente inclusa nel jar) con dentro i file delle dll.

Verifica il nome che ti sputa la mapLibrary:

System.out.println(libraryName);

e controlla che il tuo jar contenga effettivamente il file:

/natives/nome mappato

Nota che per eseguire l'applicazione "senza il jar" (cioè con il run dell'ide) occorre che il classpath contenga la cartella di prima (in netbeans basta dichiararla come un package del progetto e credo lo stesso valga per Eclipse ma non sono sicurissimo).

topix93
22-05-2012, 18:48
Secondo te come mai quando avvio il jar (facendo java -jar mio.jar) ricevo questo errore:

Exception in thread "main" java.lang.NoClassDefFoundError: net/contentobjects/jn
otify/JNotifyListener
Caused by: java.lang.ClassNotFoundException: net.contentobjects.jnotify.JNotifyL
istener
at java.net.URLClassLoader$1.run(Unknown Source)
at java.security.AccessController.doPrivileged(Native Method)
at java.net.URLClassLoader.findClass(Unknown Source)
at java.lang.ClassLoader.loadClass(Unknown Source)
at sun.misc.Launcher$AppClassLoader.loadClass(Unknown Source)
at java.lang.ClassLoader.loadClass(Unknown Source)
Could not find the main class: kjb.Main. Program will exit.

Sono sicuro che sia tutto a posto, ho aggiunto il jar e le librerie al classpath, non so piu dove sbattere la testa

PGI-Bis
22-05-2012, 18:58
Apri mio.jar (con uno zipper) vai in META-INF e apri il file MANIFEST.MF. Controlla che ci sia una linea "Class-Path" e che contenga un riferimento alla libreria jnotify.jar

topix93
22-05-2012, 19:24
Non c'è una linea class-path... cosa ci aggiungo?

PGI-Bis
22-05-2012, 19:40
devi metterci un riferimento al percorso della libreria jnotify.jar, relativo alla posizione del file jar.

Ad esempio puoi dire:

Class-Path: libraries/jnotify.jar

e la jvm si aspetterà che nella cartella dove si trova mio.jar ci sia una cartella "libraries" con dentro il jnotify.jar

topix93
22-05-2012, 19:48
nel manifest ho messo:

Manifest-Version: 1.0
Sealed: true
Main-Class: kjb.Main
Class-Path: lib/jnotify.jar


nel jar c'è la cartella lib con dentro il jar però mi da sempre lo stesso errore

PGI-Bis
22-05-2012, 20:57
jnotify.jar deve stare fuori da mio.jar (non come la dll, deve proprio essere fuori dall'archivio, nella sua cartella "lib")

topix93
22-05-2012, 21:59
ora pero ho di nuovo questo errore
Error loading library, java.library.path=C:\Windows\system32;C:\Windows\Sun\Java
\bin;C:\Windows\system32;C:\Windows;C:\Windows\system32;C:\Windows;C:\Windows\Sy
stem32\Wbem;C:\Windows\System32\WindowsPowerShell\v1.0\;C:\Program Files (x86)\M
icrosoft SQL Server\100\Tools\Binn\;C:\Program Files\Microsoft SQL Server\100\To
ols\Binn\;C:\Program Files\Microsoft SQL Server\100\DTS\Binn\;C:\Go\bin;.
Exception in thread "main" java.lang.UnsatisfiedLinkError: no jnotify_64bit in j
ava.library.path
at java.lang.ClassLoader.loadLibrary(Unknown Source)
at java.lang.Runtime.loadLibrary0(Unknown Source)
at java.lang.System.loadLibrary(Unknown Source)
at net.contentobjects.jnotify.win32.JNotify_win32.<clinit>(Unknown Sourc
e)
at net.contentobjects.jnotify.win32.JNotifyAdapterWin32.<init>(Unknown S
ource)
at sun.reflect.NativeConstructorAccessorImpl.newInstance0(Native Method)

at sun.reflect.NativeConstructorAccessorImpl.newInstance(Unknown Source)

at sun.reflect.DelegatingConstructorAccessorImpl.newInstance(Unknown Sou
rce)
at java.lang.reflect.Constructor.newInstance(Unknown Source)
at java.lang.Class.newInstance0(Unknown Source)
at java.lang.Class.newInstance(Unknown Source)
at net.contentobjects.jnotify.JNotify.<clinit>(Unknown Source)
at Main.main(Main.java:33)


tieni conto che prima funzionava sena fare System.load perchè Quando avevo aggiunto il jar al class-path avevo specificato come "Native Library Location" la cartella natives con dentro le dll... adesso ho anche provato tirando fuori dal jar la cartella natives ma non funziona

PGI-Bis
22-05-2012, 22:09
Forse è un conflitto di nomi. Prova a chiamare la cartella (esterna) in cui si trova il jar jnotify.jar "libraries" anzichè "lib"

topix93
22-05-2012, 22:19
niente

PGI-Bis
22-05-2012, 22:28
Il tuo PC ha bisogno di un esorcista.

Usa uno script di avvio:

java -cp .;mio.jar;jnotify.jar -Djava.library.path=cartella che contiene la dll kjb.Main

topix93
22-05-2012, 22:34
Mi dice impossibile trovare o caricare la classe principale Main

topix93
22-05-2012, 22:44
se riesci magari prova tu, ti lascio i miei file cosa magari riesci a capire meglio la situazione... io non sono così esperto come te.

https://rapidshare.com/files/2955352201/Desktop.rar

PGI-Bis
23-05-2012, 07:44
java -cp -;aaa.jar;.\libraries\jnotify.jar -Djava.library.path=.\natives\ Main

Il programma non fa l'autoestrazione delle dll quindi devi usare la linea di comando.

topix93
23-05-2012, 07:57
Ma questa cosa da linea di comando dovrei farla tutte le volte?

PGI-Bis
23-05-2012, 08:18
Usi uno script di avvio, di solito un file batch che accompagna il tuo programma (tutti i percorsi sono relativi alla directory di lavoro). L'utente esegue lo script e il programma parte.

topix93
23-05-2012, 20:43
thanks, ora e tutto a posto