Как я программно компилирую и инстанцирую класса Java? [дубликат]

64
задан BalusC 23 February 2011 в 12:04
поделиться

2 ответа

Как загрузить класс Java, который не скомпилирован?

Сначала его нужно скомпилировать. Это можно сделать программно с помощью javax.tools API. Для этого нужно только, чтобы JDK был установлен на локальной машине поверх JRE.

Вот базовый начальный пример (оставив в стороне очевидную обработку исключений):

// Prepare source somehow.
String source = "package test; public class Test { static { System.out.println(\"hello\"); } public Test() { System.out.println(\"world\"); } }";

// Save source in .java file.
File root = new File("/java"); // On Windows running on C:\, this is C:\java.
File sourceFile = new File(root, "test/Test.java");
sourceFile.getParentFile().mkdirs();
Files.write(sourceFile.toPath(), source.getBytes(StandardCharsets.UTF_8));

// Compile source file.
JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
compiler.run(null, null, null, sourceFile.getPath());

// Load and instantiate compiled class.
URLClassLoader classLoader = URLClassLoader.newInstance(new URL[] { root.toURI().toURL() });
Class<?> cls = Class.forName("test.Test", true, classLoader); // Should print "hello".
Object instance = cls.newInstance(); // Should print "world".
System.out.println(instance); // Should print "test.Test@hashcode".

Что получается

hello
world
test.Test@ab853b

Дальнейшее использование будет более простым, если эти классы реализуют определенный интерфейс, который уже есть в classpath.

SomeInterface instance = (SomeInterface) cls.newInstance();

В противном случае вам придется задействовать Reflection API для доступа и вызова (неизвестных) методов/полей.


По этому поводу и без связи с фактической проблемой:

properties.load(new FileInputStream(new File("ClassName.properties")));

Позволить java.io.File полагаться на текущий рабочий каталог - это рецепт проблем с переносимостью. Не делайте этого. Поместите этот файл в classpath и используйте ClassLoader#getResourceAsStream() с относительным к classpath путем.

properties.load(Thread.currentThread().getContextClassLoader().getResourceAsStream("ClassName.properties"));
132
ответ дан 24 November 2019 в 15:47
поделиться

Ваш закомментированный код корректен, если вы знаете, что класс имеет публичный no-arg конструктор. Вы просто должны привести результат, так как компилятор не может знать, что класс на самом деле реализует IDynamicLoad. Итак:

   IDynamicLoad newClass = (IDynamicLoad) Class.forName(class_name).newInstance();

Конечно, чтобы это сработало, класс должен быть скомпилирован и находиться в classpath.

Если вы хотите динамически скомпилировать класс из исходного кода, то это совсем другое дело.

5
ответ дан 24 November 2019 в 15:47
поделиться
Другие вопросы по тегам:

Похожие вопросы: