引言
在JAVA语言里面,类型得加载,连接和初始化过程都是在程序运行期间完成的,这种策略虽然会令类加载时稍微增加一些性能开销,但是给JAVA应用程序提供了高度的灵活性。
比如 在面向接口编程,运行时我才知道我具体采用哪个实现类去加载 (JAVA反射)。
比如 在运行时从网络或者其他地方加载一个二进制流作为代码的一部分。
类加载过程分为三个阶段:
一 加载
这一阶段通过类加载器进行
1.从各个地方获取类的二进制字节流,比如从war包,jar包,各种文件获取。
2.将字节流中的静态存储结构转化为方法区的运行时数据结构。
2.在内存中生成一个java.lang.class对象,作为这个类的访问入口。
代码如下
InputStream in = this.getClass().getClassLoader().getResourceAsStream(path);
二 连接
具体分为三个阶段:
1.验证 保证字节流中包含的文件符合当前虚拟机的要求,并且不会危害到虚拟机的安全。
如不和规范会抛出java.lan.VerifyError异常。
2.准备
为静态变量( static修饰的变量)在方法区中分配内存。
3.解析
将常量池中的符号引用替换为直接引用的过程。就是解析类或接口中的方法字段,如果没有就会抛出相应异常。
三 初始化
执行类构造器的方法,其不同与实例构造器方法,不需要显示的调用。
双亲委派模型
类加载器之间的层级关系被称为双亲委派模型,双亲委派模型的工作过程是:如果一个类加载器收到了类加载的请求,它首先不会自己去尝试加载这个类,而是把这个请求委派给父类加载器去完成,每一个层次都是如此,只有当父加载器反馈自己无法加载请求时,子加载器才会尝试自己去加载。
这样的好处是 保证了整个JAVA的类被不同类加载器 加载出来都是 同一个。
实现方法 伪代码
Boolean c= findLoadedClass()..
if(c){//未被加载
try{
if(父类非空){
parent.loadClass..
}esle{
bootstrapClass.loadClass..
}
}catch{
//父加载器记载失败
this.loadClass()...
}
}else