1. 程式人生 > >自定義ClassLoader,編譯並載入外部java檔案

自定義ClassLoader,編譯並載入外部java檔案

/**

* java檔案: Test.java,  package: com.chenyf.entity,  所在目錄:F:/today/javadir/src/main/java/com/chenyf/entity/Test.java

*/

/**
* 編譯java檔案,並載入其Class的工具
* dirPath: 不含包目錄的java檔案所在目錄, F:/today/javadir/
* pakagePath: 包路徑,com.chenyf.entity
*/

public static Class getClassFromJavaFile(String dirPath, String pakagePath) {
    if  (StringUtils.isBlank(dirPath) || StringUtils.isBlank(pakagePath))  //校驗引數是否為空
return null;

String pakageDir = pakagePath.replaceAll("\\.","/");  // 將路徑中的 . 替換為 / , 替換後的pakageDir = com/chenyf/entity
String filePath = dirPath.concat( "/src/main/java/" ).concat( pakageDir ).concat(".java");  // src/main/java為java檔案的特定目錄

//編譯
JavaCompiler javac = ToolProvider.getSystemJavaCompiler();
int compilationResult = javac.run(null, null, null, filePath);

if (compilationResult != 0)  //compilationResult == 0,說明編譯成功,在Java檔案的同目錄下會生成相應的class檔案
throw new IllegalArgumentException("編譯失敗");

Class<?> clazz = null;
try {

MyClassLoader loader = new MyClassLoader(dirPath);  //使用自定義ClassLoader
clazz = loader.findClass(pakagePath);

} catch (Exception e) {
e.printStackTrace();
throw e;
}
return clazz
}

/**
*自定義ClassLoader
*/
private static final class MyClassLoader extends ClassLoader {

private String classDir;  // 檔案目錄,例如:"file:/today/javadir/src/main/java/"
    @Override
public Class<?> findClass(String name) {
String realPath = classDir + name.replace(".","/") + ".class";  //class檔案的真實路徑
byte[] cLassBytes = null;
Path path = null;

try {

path = Paths.get(new URI(realPath));  
cLassBytes = Files.readAllBytes(path);

} catch (IOException | URISyntaxException e) {
e.printStackTrace();
}
Class clazz = defineClass(name, cLassBytes, 0, cLassBytes.length);  //呼叫父類的defineClass方法
return clazz;
}

public MyClassLoader(String classDir) {
this.classDir = "file:/".concat(classDir).concat("/src/main/java/");  //拼接 “file:/”字首
}

}