这文章主要让你们在开发的时候可以加载不一样版本类型的class。html
在不少开发的时候,咱们须要到某个jar包的某个功能,或因为升级,或其余缘由,并不老是向后兼容的,这时候咱们就应该须要支持多个不一样版本的库。 使用java自带的的加载器,并不能作到这个要求,java自带的 "loadClass"只加载一次,以后返回的就是类的引用了。java
使用其余的 ClassLoader 来加载所需jar; 下面是两种实现方法:ide
URLClassLoader 这个类容许你经过URL加载jar或者为你的类指定一个具体的目录this
example:url
URLClassLoader clsLoader = URLClassLoader.newInstance(new URL[] {new URL("file:/C://Test/test.jar")}); Class cls = clsLoader.loadClass("test.Main"); Method method = cls.getMethod("main", String[].class); String[]params = new String[2]; method.invoke(null, (Object) params);
在java的类加载器(包括:URLClassLoader)会首先要求从他们的类中加载,你这里须要自定一个类处理器来指定路径,知足你的特殊需求。 下面是一个自定义类加载器的示例:spa
import java.net.URL; import java.net.URLClassLoader; import java.util.List; public class CustomClassLoader extends ClassLoader { private ChildClassLoader childClassLoader; public CustomClassLoader(List<URL> classpath) { super(Thread.currentThread().getContextClassLoader()); URL[] urls = classpath.toArray(new URL[classpath.size()]); childClassLoader = new ChildClassLoader( urls, new DetectClass(this.getParent()) ); } @Override protected synchronized Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException { try { return childClassLoader.findClass(name); } catch( ClassNotFoundException e ) { return super.loadClass(name, resolve); } } private static class ChildClassLoader extends URLClassLoader { private DetectClass realParent; public ChildClassLoader( URL[] urls, DetectClass realParent ) { super(urls, null); this.realParent = realParent; } @Override public Class<?> findClass(String name) throws ClassNotFoundException { try { Class<?> loaded = super.findLoadedClass(name); if( loaded != null ) return loaded; return super.findClass(name); } catch( ClassNotFoundException e ) { return realParent.loadClass(name); } } } private static class DetectClass extends ClassLoader { public DetectClass(ClassLoader parent) { super(parent); } @Override public Class<?> findClass(String name) throws ClassNotFoundException { return super.findClass(name); } } }