java 类加载器-基础
类加载机制
- 类加载器ClassLoader。
– 负责查找,加载,校验字节码的应用程序。
– java.lang.ClassLoader主要方法:
- load(String className),根据名字加载一个类。
- defineClass(String name, byte[] b, int off, int len),将一个字节流定义为一个类。
- findClass(String name),查找一个类。
- findLoadedClass(String name),在已加载的类中,查找一个类。
- 成员变量ClassLoader parent。
- JVM的四级类加载器。
- 启动类加载器(Bootstrap),加载系统类tr.jar。
- 扩展类加载器(Extension),jre/lib/ext、
- 应用类加载器(App),classpath。
- 用户定义加载器(Plugin)。
类加载器的双亲委托机制
java严格遵守双亲委托机制。
- 首先判断是否已经加载。
- 若无,找父加载器加载。
- 若再无,由当前加载器加载。
- 类会由最顶层的类加载器来加载,如果没有,则由当前类加载器加载。
- 委托是单向的,即只有下级类加载器加载的类可以访问上级类加载器加载的类。
- 双亲委托的补充:
– 执行java程序,添加虚拟机参数-Xbootclasspath/a:<path>,将类路径配置为Bootstrap等级。该类将会被JVM优先加载。
自定义类加载路径
-
自定义类加载路径:
– 弥补类搜索路径静态的不足。
– 前3个加载的路径在运行前已经确定。
– java提供URLClassLoader使得程序运行时可以修改类的加载路径。 -
URLClassLoader
– 继承于ClassLoader。
– 程序运行时可以修改类的加载路径。
– 可已从多个来源中加载类:目录,jar包,网络。
该类有两个构造函数。
URLClassLoader(URL[] urls);
//使用默认的父加载器加载给定的URLClassLoader对象。
URLClassLoader(URL[] urls, ClassLoader parent);
//利用给定的类加载器创建URLClassLoader对象。
自定义类加载器
- 继承ClassLoader类。
- 重写findClass(String className)方法。
例如:用project2下的myClassLoader类去加载project1下的hello类,并调用其say方法。
project1:
public class hello{
public void say(){
System.out.println("hello");
}
}
project2:
public class myClassLoader extends ClassLader{
public class<?> findClass(String name)throws ClassNotFoundException{
byte[] classBytes = null;
classBytes = loadClassBytes(name);
//调用defineClass产生一个类,并在VM中注册。
Class<?> c = defineClass(name, classBytes, 0, classBytes.length);
if(c == null) throw new ClassNotFoundException(name);
return c;
}
private byte loadClassBytes(name){
String cname = "paths/hello.class";
byte[] bytes = Files.readAllBytes(cname);
return bytes;
}
}
public class Test{
public static void main(String[] args) throws Exception{
ClassLoader classloader = new myClassLoader();
//调用loadClass去加载hello。
//loadClass是ClassLoader的默认方法,采用双亲原则加载类。
//loadClass找不到,就会调用findClass找。
Class<?> c = classloader.loadClass("hello");
Method m = c.getMethod("say");
Object obj = c.newInstance();
m.invoke(obj);
System.out.println(c.getClassLoader().getClass().getName());
}
}
输出结果:
hello
myClassLoader