加载
读取二进制流的一个过程,二进制流的来源可以有很多,比如:本地磁盘、反射代理、JSP的动态创建等。
加载过程中会在内存中创建一个java.lang.Class对象,分配到方法区中,作为入口
验证
定义: 验证当前的二进制流是否符合JVM虚拟机的规范
验证的内容:
- 文件格式验证,验证的是CLass文件
- 元数据验证,又可以叫做语义分析验证,说白了其实就是java语言基本规范
- 字节码验证,又可以理解为java类的方法和逻辑的校验
- 符合引用验证,说白了其实就是对除本类之外的其他类引用的校验。NoSuchFiledError,NoSuchMethodError 等常见错误
准备
定义: 正式为类变量分配内存和设置初始值的阶段
注意:1. 正常情况下类的初始值都是数据类型的零值
2.如果一个类变量被final 修饰符修饰的话,这时候会被分配到常量池,且值已经赋值好
解析
将常量池中的符号引用转变为直接引用的过程
初始化
定义 是指执行类构造器 < clinit() >方法的过程
< clinit() >的解释:
- < clinit() >方法是类中所有类变量的赋值动作和静态代码块语句合成的
- < clinit() > 方法和类的构造函数不同,子类的< clinit() >方法会自动加载其父类的< clinit() >方法
- 如果一个类中没有静态语句块和类变量的赋值操作,则不会生成< clinit() >
- 再多线程的情况下,一个类只会初始化一次
类的双亲委派模型
类加载器: 实际上是一段代码块,代码块实现的功能是“通过一个类的全限定名来获取描述此类的二进制字节流”这个动作放到Java虚拟机外部实现,以便让应用程序自己决定如何去获取所需要的类“
类加载器的分类
- 启动类加载器(bootstrap):是用本地代码实现的类装入器,它负责将 <Java_Runtime_Home>/lib下面的类库加载到内存中(比如rt.jar)
- 扩展类加载器(Extendsion):是由 Sun 的ExtClassLoader(sun.misc.Launcher$ExtClassLoader)实现的。它负责将< Java_Runtime_Home >/lib/ext或者由系统变量 java.ext.dir指定位置中的类库加载到内存中。开发者可以直接使用标准扩展类加载器。
- ** 系统类加载器**是由 Sun 的 AppClassLoader(sun.misc.Launcher$AppClassLoader)实现的。它负责将系统类路径(CLASSPATH)中指定的类库加载到内存中。开发者可以直接使用系统类加载器。
[图片上传失败...(image-6cb210-1512205433219)]
双亲委派模型
某个特定的类加载器在接到加载类的请求时,首先将加载类的任务委托给父类加载器,依次递归,如果父类加载器可以完成加载任务,则返回;如果不能,则自己完成加载
委托机制的意义 防止java内存中存在多份同样的字节码
测试题
package com.zzjmay;
public class TestJVM2 {
public static int k = 0;
public static TestJVM2 t1 = new TestJVM2("t1");
public static int i = print("i");
public static int n = 99;
public int j = print("j");
{
print("构造块");
}
static {
print("静态块");
}
public TestJVM2(String str){
System.out.println((++k)+":"+str+"i="+i+"n="+n);
++n;
++i;
}
public static int print(String str){
System.out.println((++k) + ":" + str + " i=" + i + " n=" + n);
++i;
return ++n;
}
public static void main(String[] args) {
TestJVM2 testJVM2 = new TestJVM2("init");
}
}