2009-12-31 3 views
113

Je ne suis pas en face d'un IDE en ce moment, je regarde simplement les spécifications de l'API.Déterminez le fichier JAR d'une classe à partir de

CodeSource src = MyClass.class.getProtectionDomain().getCodeSource(); 
if (src != null) { 
    URL jar = src.getLocation(); 
} 

Je souhaite déterminer le fichier JAR d'une classe. Est-ce la façon de le faire?

+0

Est-il possible de le faire à partir de la console? Quelque chose comme 'java -findjar -cp /some/path/with/libs/*.jar my.java.Class' ->' my.jar'. – kub1x

Répondre

148

Oui. Cela fonctionne pour toutes les classes sauf les classes chargées par le chargeur de classe bootstrap. L'autre façon de déterminer est:

Class klass = String.class; 
URL location = klass.getResource('/' + klass.getName().replace('.', '/') + ".class"); 

Comme méthode notnoop a fait remarquer getProtectionDomain().getCodeSource().getLocation() renvoie l'emplacement du fichier de classe elle-même. Par exemple:

jar:file:/jdk/jre/lib/rt.jar!/java/lang/String.class 
file:/projects/classes/pkg/MyClass$1.class 

La méthode klass.getResource() retourne l'emplacement du fichier jar ou CLASSPATH

file:/Users/home/java/libs/ejb3-persistence-1.0.2.GA.jar 
file:/projects/classes 
+0

Cela fait des hypothèses sur le mappage du nom de classe au fichier de classe. Cela fonctionnera-t-il correctement pour les cours anonymes? Classes imbriquées –

+1

Ceci indique l'URL de la classe et non le pot lui-même. L'URL doit être analysée pour trouver le fichier JAR. – notnoop

+0

@notnoop. J'ai clarifié la réponse. –

11

Commander le LiveInjector.findPathJar() de Lombok Patcher LiveInjector.java. Notez qu'il existe des cas spéciaux où le fichier ne vit pas réellement dans un pot, et vous pourriez vouloir changer cela.

/** 
* If the provided class has been loaded from a jar file that is on the local file system, will find the absolute path to that jar file. 
* 
* @param context The jar file that contained the class file that represents this class will be found. Specify {@code null} to let {@code LiveInjector} 
*    find its own jar. 
* @throws IllegalStateException If the specified class was loaded from a directory or in some other way (such as via HTTP, from a database, or some 
*        other custom classloading device). 
*/ 
public static String findPathJar(Class<?> context) throws IllegalStateException { 
    if (context == null) context = LiveInjector.class; 
    String rawName = context.getName(); 
    String classFileName; 
    /* rawName is something like package.name.ContainingClass$ClassName. We need to turn this into ContainingClass$ClassName.class. */ { 
     int idx = rawName.lastIndexOf('.'); 
     classFileName = (idx == -1 ? rawName : rawName.substring(idx+1)) + ".class"; 
    } 

    String uri = context.getResource(classFileName).toString(); 
    if (uri.startsWith("file:")) throw new IllegalStateException("This class has been loaded from a directory and not from a jar file."); 
    if (!uri.startsWith("jar:file:")) { 
     int idx = uri.indexOf(':'); 
     String protocol = idx == -1 ? "(unknown)" : uri.substring(0, idx); 
     throw new IllegalStateException("This class has been loaded remotely via the " + protocol + 
       " protocol. Only loading from a jar on the local file system is supported."); 
    } 

    int idx = uri.indexOf('!'); 
    //As far as I know, the if statement below can't ever trigger, so it's more of a sanity check thing. 
    if (idx == -1) throw new IllegalStateException("You appear to have loaded this class from a local jar file, but I can't make sense of the URL!"); 

    try { 
     String fileName = URLDecoder.decode(uri.substring("jar:file:".length(), idx), Charset.defaultCharset().name()); 
     return new File(fileName).getAbsolutePath(); 
    } catch (UnsupportedEncodingException e) { 
     throw new InternalError("default charset doesn't exist. Your VM is borked."); 
    } 
} 
+2

Cela semble trop compliqué pour obtenir quelque chose de très simple. Je me suis assis et j'ai essayé ce que j'ai trouvé plus tôt, et cela semble fonctionner. Je voulais juste une validation. –

+0

Eh bien. Votre code ne gère pas les fichiers dans les chemins bootclass, et la solution de Chandra renvoie l'URL au fichier et non au fichier jar, vous devrez donc analyser le chemin pour trouver le fichier jar. – notnoop

0
private String resourceLookup(String lookupResourceName) { 



    try { 

     if (lookupResourceName == null || lookupResourceName.length()==0) { 
      return ""; 
     } 
     // "/java/lang/String.class" 

     // Check if entered data was in java class name format 
     if (lookupResourceName.indexOf("/")==-1) { 
      lookupResourceName = lookupResourceName.replaceAll("[.]", "/"); 
      lookupResourceName = "/" + lookupResourceName + ".class"; 
     } 

     URL url = this.getClass().getResource(lookupResourceName); 
     if (url == null) { 
      return("Unable to locate resource "+ lookupResourceName); 

     } 

     String resourceUrl = url.toExternalForm(); 

     Pattern pattern = 
      Pattern.compile("(zip:|jar:file:/)(.*)!/(.*)", Pattern.CASE_INSENSITIVE); 

     String jarFilename = null; 
     String resourceFilename = null; 
     Matcher m = pattern.matcher(resourceUrl); 
     if (m.find()) { 
      jarFilename = m.group(2); 
      resourceFilename = m.group(3); 
     } else { 
      return "Unable to parse URL: "+ resourceUrl; 

     } 

     if (!jarFilename.startsWith("C:")){ 
      jarFilename = "/"+jarFilename; // make absolute path on Linux 
     } 

     File file = new File(jarFilename); 
     Long jarSize=null; 
     Date jarDate=null; 
     Long resourceSize=null; 
     Date resourceDate=null; 
     if (file.exists() && file.isFile()) { 

      jarSize = file.length(); 
      jarDate = new Date(file.lastModified()); 

      try { 
       JarFile jarFile = new JarFile(file, false); 
       ZipEntry entry = jarFile.getEntry(resourceFilename); 
       resourceSize = entry.getSize(); 
       resourceDate = new Date(entry.getTime()); 
      } catch (Throwable e) { 
       return ("Unable to open JAR" + jarFilename + " "+resourceUrl +"\n"+e.getMessage()); 

      } 

      return "\nresource: "+resourceFilename+"\njar: "+jarFilename + " \nJarSize: " +jarSize+" \nJarDate: " +jarDate.toString()+" \nresourceSize: " +resourceSize+" \nresourceDate: " +resourceDate.toString()+"\n"; 


     } else { 
      return("Unable to load jar:" + jarFilename+ " \nUrl: " +resourceUrl); 

     } 
    } catch (Exception e){ 
     return e.getMessage(); 
    } 


} 
+0

Le code ci-dessus trouvera n'importe quelle ressource sur le chemin. Si dans un bocal, le bocal affiche la taille et la date du bocal ainsi que la taille et la date de la ressource dans le bocal – Don

Questions connexes