Добавление файлов в java classpath во время выполнения

Можно ли добавить файл (не обязательно файл jar) в java classpath во время выполнения. В частности, файл уже присутствует в пути к классам, я хочу, могу ли я добавить измененную копию этого файла в путь к классам.

Спасибо,

Ответ 1

Вы можете добавлять папки или файлы jar только в загрузчик классов. Поэтому, если у вас есть один файл класса, сначала нужно поместить его в соответствующую структуру папок.

Здесь - довольно уродливый взлом, который добавляет SystemClassLoader во время выполнения:

import java.io.IOException;
import java.io.File;
import java.net.URLClassLoader;
import java.net.URL;
import java.lang.reflect.Method;

public class ClassPathHacker {

private static final Class[] parameters = new Class[]{URL.class};

public static void addFile(String s) throws IOException {
   File f = new File(s);
   addFile(f);
}//end method

public static void addFile(File f) throws IOException {
   addURL(f.toURL());
}//end method


public static void addURL(URL u) throws IOException {

  URLClassLoader sysloader = (URLClassLoader) ClassLoader.getSystemClassLoader();
  Class sysclass = URLClassLoader.class;

  try {
     Method method = sysclass.getDeclaredMethod("addURL", parameters);
     method.setAccessible(true);
     method.invoke(sysloader, new Object[]{u});
  } catch (Throwable t) {
     t.printStackTrace();
     throw new IOException("Error, could not add URL to system classloader");
  }//end try catch

   }//end method

}//end class

Отражение необходимо для доступа к защищенному методу addURL. Это может привести к сбою, если есть SecurityManager.

Ответ 2

Попробуйте этот вариант для размера.

private static void addSoftwareLibrary(File file) throws Exception {
    Method method = URLClassLoader.class.getDeclaredMethod("addURL", new Class[]{URL.class});
    method.setAccessible(true);
    method.invoke(ClassLoader.getSystemClassLoader(), new Object[]{file.toURI().toURL()});
}

Это редактирует загрузчик системного класса, чтобы включить данную библиотеку jar. Это довольно уродливо, но оно работает.

Ответ 3

Как я это сделал, используя мой собственный загрузчик классов

URLClassLoader urlClassLoader = (URLClassLoader) ClassLoader.getSystemClassLoader();
DynamicURLClassLoader dynalLoader = new DynamicURLClassLoader(urlClassLoader);

И создайте следующий класс:

public class DynamicURLClassLoader extends URLClassLoader{

public DynamicURLClassLoader(URLClassLoader classLoader) {
    super(classLoader.getURLs());

}

@Override
public void addURL(URL url) {
    super.addURL(url);
}

}

Работает без отражения

Ответ 4

Вы можете попробовать java.net.URLClassloader с URL-адресом папки /jar, где находится обновленный класс, и использовать его вместо загрузчика классов по умолчанию при создании нового потока.

Ответ 5

да, вы можете. он должен будет находиться в структуре пакета в отдельном каталоге из остальной части вашего скомпилированного кода, если вы хотите его изолировать. тогда вы просто поместите свой базовый каталог в начало пути класса в командной строке.

Ответ 6

Да, я считаю, что это возможно, но вам, возможно, придется реализовать свой собственный загрузчик классов. Я никогда не делал этого, но это тот путь, на который я, вероятно, посмотрю.

Ответ 7

Мое решение:

File jarToAdd = new File("/path/to/file");

new URLClassLoader(((URLClassLoader) ClassLoader.getSystemClassLoader()).getURLs()) {
    @Override
    public void addURL(URL url) {
         super.addURL(url);
    }
}.addURL(jarToAdd.toURI().toURL());