Décès inopiné (et sans aucun log !) de la JVM
Bonjour,
Le code de mon application est réparti dans une dizaine de .jar et utilise une quinzaine de jars de librairies externes.
Mon application tourne parfaitement si je définis une variable d'environnement CLASSPATH et que je la lance par
java -classpath %CLASSPATH% mon.package.MaClasse
Toutefois, ayant eu des erreurs dues à la longueur de la variable d'environnement sur certains systèmes (WinNT, MacOSX) j'ai essayé de réaliser un "loader", un peu comme le start.jar de Tomcat.
Le problème, c'est que quand j'utilise mon "loader", l'application se lance puis meurt subitement, sans prévenir, après quelques secondes d'exécution. Le plus étrange, c'est qu'il n'y a rien sur l'erreur qui provoque ça, ni dans les logs (j'utilise log4j), ni sur la console. Je ne retrouve même pas de fichier d'erreur indiquant un arrêt brutal de la JVM (violation d'accès ou autre).
J'ai ajouté un UncaughtExceptionHandler et un ShutdownHook pour essayer de voir ce qui se passe, mais rien à faire : toujours aucun log.
J'ai testé sous Windows 2000, Windows XP et MacOSX, avec les mêmes résultats à chaque fois (ràs si lancement par script, décès brutal si lancement via le "loader").
Voyez-vous une explication à cette abscence de traces sur la raison de cet arrêt ? Une idée sur comment identifier le problème ? Une bourde énorme dans mon code (ci-dessous) ?
la classe principale (le "main" du launcher)
Code:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34
|
package mon.test;
import java.lang.reflect.Method;
public class Demarrer
{
public static void main(String[] args)
{
// Définition du classpath
System.setProperty("java.class.path",ClassPathBuilder.getStringValue());
// Chargement d'un Classloader avec ce classpath
ClassLoader loader = ClassPathBuilder.getClassLoader();
Thread.currentThread().setContextClassLoader(loader);
try
{
// Chargement de la classe principale
Class mon_main = loader.loadClass("mon.package.MaClasse");
// Lancement de l'application
Method main = mon_main.getDeclaredMethod("main", new Class[] { args.getClass() });
main.invoke(null, new Object[] { args });
}
catch(Throwable t)
{
System.err.println("Echec du lancement");
t.printStackTrace(System.err);
}
System.exit(0);
}
} |
Et la classe ClassPathBuilder
Code:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156
|
package mon.test;
import java.io.File;
import java.net.MalformedURLException;
import java.net.URL;
import java.net.URLClassLoader;
import java.util.HashMap;
import java.util.Iterator;
import java.util.ResourceBundle;
public class ClassPathBuilder {
/**
* Les librairies à ajouter au Classpath sous tous les systèmes
*/
private static String[] requiredLibs;
/**
* Les librairies à rajouter au classpath quand on est sous Windows
*/
private static String[] win32Libs;
/**
* Les librairies à rajouter au classpath quand on est sous Mac
*/
private static String[] macLibs;
static
{
// Initialisation des librairies
ResourceBundle rb = ResourceBundle.getBundle("classpath");
int reqLibCount = Integer.parseInt(rb.getString("requiredLibs.count"));
requiredLibs = new String[reqLibCount];
for (int i = 0; i< reqLibCount; i++)
{
requiredLibs[i] = rb.getString("requiredLib." + i);
}
int win32LibCount = Integer.parseInt(rb.getString("win32Libs.count"));
win32Libs = new String[win32LibCount];
for (int i = 0; i< win32LibCount; i++)
{
win32Libs[i] = rb.getString("win32Lib." + i);
}
int macLibCount = Integer.parseInt(rb.getString("macLibs.count"));
macLibs = new String[macLibCount];
for (int i = 0; i< macLibCount; i++)
{
macLibs[i] = rb.getString("macLib." + i);
}
}
private static final boolean isOnWindows = (System.getProperty("os.name").indexOf("Win") != -1);
private static final boolean isOnMac = (System.getProperty("os.name").indexOf("Mac") != -1);
private static HashMap<String,File> foundLibs = null;
private static File trouverLibrairie(String nomLib)
{
File fichierLib = new File(nomLib);
if (fichierLib.exists())
{
return fichierLib;
}
else
{
fichierLib = new File("./lib/" + nomLib);
if (fichierLib.exists())
{
return fichierLib;
}
}
System.err.println(nomLib + " requise mais non trouvee");
return null;
}
private static HashMap<String,File> getFoundLibs()
{
if (foundLibs != null) return foundLibs;
foundLibs = new HashMap<String,File>();
int numLibs = requiredLibs.length;
if (isOnWindows) numLibs += win32Libs.length;
if (isOnMac) numLibs += macLibs.length;
// Chargement des librairies communes
for (int i=0; i < requiredLibs.length; i++)
{
File f = trouverLibrairie(requiredLibs[i]);
if (f != null) foundLibs.put(requiredLibs[i], f);
}
// Et des librairies Windows s'il y a lieu
if (isOnWindows)
{
for (int i=0 ; i < win32Libs.length ; i++)
{
File f = trouverLibrairie(win32Libs[i]);
if (f != null) foundLibs.put(win32Libs[i], f);
}
}
// Et des librairies Mac s'il y a lieu
if (isOnMac)
{
for (int i=0 ; i < macLibs.length ; i++)
{
File f = trouverLibrairie(macLibs[i]);
if (f != null) foundLibs.put(macLibs[i], f);
}
}
return foundLibs;
}
/**
* @return Le ClassLoader initialisé avec le classpath convenable pour l'application
*
*/
public static ClassLoader getClassLoader()
{
URL[] urls = new URL[getFoundLibs().size()];
Iterator<File> fichiers = getFoundLibs().values().iterator();
for (int i=0; i < urls.length; i++)
{
try
{
urls[i] = fichiers.next().toURL();
} catch (MalformedURLException e) {}
}
// Et on crée le ClassLoader avec les jars en paramètres
ClassLoader parent = Thread.currentThread().getContextClassLoader();
if (parent == null) {
parent = ClassPathBuilder.class.getClassLoader();
}
if (parent == null) {
parent = ClassLoader.getSystemClassLoader();
}
return new URLClassLoader(urls, parent);
}
public static String getStringValue()
{
StringBuilder sb = new StringBuilder();
Iterator<File> fichiers = getFoundLibs().values().iterator();
if (fichiers.hasNext()) {
sb.append( fichiers.next().getPath() );
}
while (fichiers.hasNext()) {
sb.append(File.pathSeparatorChar).append( fichiers.next().getPath() );
}
return sb.toString();
}
} |
Merci