繼承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")); }