繼承ClassLoader並且重寫findClass方法就可以自定義一個類加載器,具體什么是類加載器以及類加載器的加載過程與順序下次再說,下面給出一個小demo
首先定義一個類,比如MyTest,並且將其編譯成class文件,然后放到一個指定的文件夾下面,其中文件夾的最后幾層就是它的包名,這里我將這個編譯好的類放到 : /Users/allen/Desktop/cn/lijie/bug .class
package cn.lijie;
public class bugnew {
public void shownew() {
System.out.println("shownew test!");
}
}
package com.infosec.manager.controller.login;
import java.lang.reflect.Method;
import java.net.URI;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
public class bug extends ClassLoader {
public void show() {
System.out.println("show test!");
}
public void shows() {
String myPath = "file:///D:/infosec/netauth/accountsync/bugnew.class";
System.out.println(myPath);
byte[] cLassBytes = null;
Path path = null;
try {
path = Paths.get(new URI(myPath));
cLassBytes = Files.readAllBytes(path);
Class clazz = defineClass("cn.lijie.bugnew", cLassBytes, 0, cLassBytes.length);
Object obj = clazz.newInstance();
Method method = clazz.getMethod("shownew");
method.invoke(obj);
}catch (Exception e) {
}
}
}
自定義的類加載器:
package com.infosec.manager.controller.login;
import java.io.IOException;
import java.net.URI;
import java.net.URISyntaxException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
public class MyClassLoader extends ClassLoader {
protected Class<?> findClass() {
String myPath = "file:///D:/infosec/netauth/accountsync/bug.class";
byte[] cLassBytes = null;
Path path = null;
try {
path = Paths.get(new URI(myPath));
cLassBytes = Files.readAllBytes(path);
} catch (IOException | URISyntaxException e) {
e.printStackTrace();
}
Class clazz = defineClass("com.infosec.manager.controller.login.bug", cLassBytes, 0, cLassBytes.length);
return clazz;
}
}
測試的主函數:
package com.infosec.manager.controller.login;
import java.lang.reflect.Method;
public class TestController {
public static void main(String[] args) throws ClassNotFoundException {
MyClassLoader loader = new MyClassLoader();
Class<?> aClass = loader.findClass();
try {
Object obj = aClass.newInstance();
Method method = aClass.getMethod("shows");
method.invoke(obj);
} catch (Exception e) {
e.printStackTrace();
}
}
}
執行主函數,調用外部class的show方法:

判斷class<?>[] 是否包含某個元素 instanceof
private boolean guessInterface(Object object) {
if (object instanceof SynchAccountApi) {
return true;
}
if (object instanceof SyncUsersFromAppInterface) {
return true;
}
Class<?>[] allInterfaces = ClassUtils.getAllInterfaces(object);
return Arrays.stream(allInterfaces).anyMatch(aClass -> aClass.getName().contains("SynchAccountApi"));
}
