У меня есть статический класс (Foo) и главный класс (Main)Java: как перезапустить статический класс?
См Main.java:
public class Main {
public static void main(String[] args) {
System.out.println(Foo.i); // 0
Foo.i++;
System.out.println(Foo.i); // 1
// restart Foo here
System.out.println(Foo.i); // 1 again...I need 0
}
}
См Foo.java:
public class Foo {
public static int i = 0;
}
Есть ли способ перезапустить или сбросить статический класс?
Примечание: Мне нужно это, потому что я тестирую статический класс с помощью jUnit, и мне нужно очистить параметры перед вторым тестом.
EDIT
ПОЧТИ РЕШЕНИЕ:
Используя StanMax ответ, я могу это:
Main.java
public class Main {
public static void main(String[] args) throws Exception {
test();
test();
}
public static void test() throws Exception {
System.out.println("\ntest()");
MyClassLoader myClassLoader = new MyClassLoader();
Class<?> fooClass = myClassLoader.loadClass(Foo.class.getCanonicalName());
Object foo = fooClass.newInstance();
System.out.println("Checking classloader: " + foo.getClass().getClassLoader());
System.out.println("GC called!");
System.gc();
}
}
MyClassLoader.java
public class MyClassLoader {
private URLClassLoader urlClassLoader;
public MyClassLoader() {
try {
URL url = new File(System.getProperty("user.dir") + "/bin/").toURL();
URL[] urlArray = {url};
urlClassLoader = new URLClassLoader(urlArray, null);
} catch (Exception e) {
}
}
public Class<?> loadClass(String name) {
try {
return (Class<?>) urlClassLoader.loadClass(name);
} catch (Exception e) {
}
return null;
}
@Override
protected void finalize() throws Throwable {
System.out.println("MyClassLoader - End.");
}
}
Foo.java
public class Foo {
public static int i = 0;
static {
System.out.println("Foo - BEGIN ---------------------------------");
}
public void finalize() throws Throwable {
System.out.println("Foo - End.");
}
}
ВЫВОД
test()
Foo - BEGIN ---------------------------------
Checking classloader: [email protected]
GC called!
MyClassLoader - End.
Foo - End.
test()
Foo - BEGIN ---------------------------------
Checking classloader: [email protected]
GC called!
MyClassLoader - End.
Foo - End.
ПРОБЛЕМА: если я бросание ниже:
Foo foo = (Foo) fooClass.newInstance();
Я получаю сообщение об ошибке:
java.lang.ClassCastException
Для этого конкретного примера просто сделайте 'Foo.i = 0;' внутри вашего метода 'Main'. – LukeH
У вас есть ClassCastException, потому что класс Foo, загруженный URLClassLoader, отличается от класса Foo, загружаемого ClassLoader, который запускает ваш код (они не ==). Даже если они имеют то же самое определение, вы не можете отбрасывать один на другой. Единственный способ использования этого трюка - если Foo реализует интерфейс FooInterface, который загружается ClassLoader, который разделяется между двумя загрузчиками классов, и в этом случае вы можете использовать FooInterface. Но это не поможет вам получить доступ к статическим полям. – NamshubWriter
NamshubWriter: Хорошо, я сделал это .... но главная проблема в том, что я использую динамический тип .... Я получаю «fooClass не может быть разрешен для типа». Возможно, что-то в Refleion API может помочь ... – Topera