Как программно скомпилировать и создать экземпляр класса java? [дубликат]
этот вопрос уже есть ответ здесь:
- Как вы динамически компилируете и загружаете внешние классы java? [дубликат] 2 ответы
У меня есть имя класса хранится в файле свойств. Я знаю, что хранилище классов будет реализовывать 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 только скомпилированный .файлы классов? Как загрузить класс 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();
в противном случае вам нужно привлечь отражение API для доступа и вызова (неизвестных) методов/полей.
что сказал и не имеет отношения к реальной проблеме:
properties.load(new FileInputStream(new File("ClassName.properties")));
сдача
java.io.File
полагаться на текущий рабочий каталог-это рецепт проблемы переносимости. Не делай этого. Поместите этот файл в classpath и используйтеClassLoader#getResourceAsStream()
С classpath-относительный путь.properties.load(Thread.currentThread().getContextClassLoader().getResourceAsStream("ClassName.properties"));
в том же духе, что и ответ BalusC, но немного более автоматическая обертка находится здесь, в этом фрагменте кода из моего дистрибутива kilim. https://github.com/kilim/kilim/blob/master/src/kilim/tools/Javac.java
Он берет список строк, содержащих источник Java, извлекает пакет и имена открытых классов / интерфейсов и создает соответствующую иерархию каталогов/файлов в каталоге tmp. Затем он запускает компилятор java на нем и возвращает список имен, classfile пары (структура ClassInfo).
помогите себе с кодом. Это лицензия MIT.
ваш прокомментированный код является правильным, если вы знаете, что класс имеет открытый конструктор no-arg. Вам просто нужно привести результат, так как компилятор не может знать, что класс фактически реализует
IDynamicLoad
. Итак:IDynamicLoad newClass = (IDynamicLoad) Class.forName(class_name).newInstance();
конечно, класс должен быть скомпилирован и на пути к классу, чтобы это работало.
Если вы хотите динамически скомпилировать класс из исходного кода, это целый другой чайник рыбы.