Как программно скомпилировать и создать экземпляр класса Java?

У меня есть имя класса хранится в файле свойств. Я знаю, что магазин классов будет реализовывать IDynamicLoad. Как создать экземпляр класса динамически?

Сейчас у меня есть

     Properties foo = new Properties();
    foo.load(new FileInputStream(new File("ClassName.properties")));
    String class_name = foo.getProperty("class","DefaultClass");
    //IDynamicLoad newClass = Class.forName(class_name).newInstance();

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

3 ответа

Решение

Как загрузить класс 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

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

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

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


Это говорит и не имеет отношения к актуальной проблеме:

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

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

properties.load(Thread.currentThread().getContextClassLoader().getResourceAsStream("ClassName.properties"));

В том же духе, что и в ответе BalusC, но в этом фрагменте кода из моего дистрибутива Kim есть немного более автоматическая оболочка. https://github.com/kilim/kilim/blob/master/src/kilim/tools/Javac.java

Он берет список строк, содержащих исходный код Java, извлекает имена пакетов и общедоступных классов / интерфейсов и создает соответствующую иерархию каталогов / файлов в каталоге tmp. Затем он запускает на нем компилятор java и возвращает список имен, пар классов (структура ClassInfo).

Помоги себе код. Это лицензия MIT.

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

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

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

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

Другие вопросы по тегам