JVM类加载机制分为五个部分:加载,验证,准备,解析,初始化,下面我们就分别来看一下这五个过程。其中加载、检验、准备、初始化和卸载这个五个阶段的顺序是固定的,而解析则未必。为了支持动态绑定,解析这个过程可以发生在初始化阶段之后。
由于Class文件并不一定由java源码编译而来,从其他途径产生的Class文件在字节码语言层面上,语义也是可以表达出来的。但是可能会载入有害字节流导致系统崩溃,如果不检查会对虚拟机造成损伤。
加载过程主要完成三件事情:
这个过程主要就是类加载器完成。
此阶段主要确保Class文件的字节流中包含的信息符合当前虚拟机的要求,并且不会危害虚拟机的自身安全。
为类变量分配内存,并将其初始化为默认值。(此时为默认值,在初始化的时候才会给变量赋值)即在方法区中分配这些变量所使用的内存空间。例如:
public static int value = 123;
此时在准备阶段过后的初始值为0而不是123;将value赋值为123的putstatic指令是程序被编译后,存放于类构造器
方法之中.特例:
public static final int value = 123;
此时value的值在准备阶段过后就是123。
把常量池内的符号引用转换为直接引用。
主要有以下四种:
初始化阶段是执行类构造器
方法的过程。
方法是由编译器自动收集类中的类变量的赋值操作和静态语句块中的语句合并而成的。虚拟机会保证
方法执行之前,父类的
方法已经执行完毕。如果一个类中没有对静态变量赋值也没有静态语句块,那么编译器可以不为这个类生成
()方法。
java中,对于初始化阶段,有且只有以下五种情况才会对要求类立刻“初始化”(加载,验证,准备,自然需要在此之前开始):
启动类加载器
(Bootstrap Class Loader):这个类加载器负责加载存放在扩展类加载器
(Extension Class Loader):这个类加载器是在类sun.misc.Launcher$ExtClassLoader中以Java代码的形式实现的。它负责加载应用程序类加载器
(Application Class Loader):这个类加载器由sun.misc.Launcher$AppClassLoader来实现。由于应用程序类加载器是ClassLoader类中的getSystem-ClassLoader()方法的返回值,所以有些场合中也称它为“系统类加载器”。它负责加载用户类路径(ClassPath)上所有的类库,在程序中可以直接使用,为程序中的默认加载器。上图中类加载器之间的层次关系称为类加载器的双亲委派模型。双亲委派模型要求除了顶层的启动类加载器之外,其余的类加载器都应当有自己的父类加载器。
*为什么要提出双亲委派模型?
其实就是为了让基础类得以正确地统一地加载。
从上面的图可以看出,如果你也定义了一个 java.lang.Object类,通过双亲委派模式是会把这个请求委托给启动类加载器,它扫描\lib目录就找到了 jdk 定义的 java.lang.Object 类来加载,所以压根不会加载你写的 java.lang.Object类,这就可以避免一些程序不小心或者有意的覆盖基础类。
所以它的优势可以归结为:
1.确保类只被加载一次,避免重复加载
2.避免核心类被任意或恶意篡改
工作过程:
如果一个类加载器收到了类加载的请求,它首先不会自己去尝试加载这个类,而是把这个请求委派给父类加载器去完成,每一层次的类加载器都如此,因此所有的加载请求应该传送到顶层的启动类加载器中,只有当父加载器反馈自己无法完成这个加载请求时即它的搜索范围内没有找到所需要的类,子加载器才会尝试自己去加载。
双亲委派模型的代码实现集中在java.lang.ClassLoader的loadClass()
方法当中。
loadClass源代码如下:
protected synchronized Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException {
//1.首先检查类是否被加载
Class c = findLoadedClass(name);
if (c == null) {
try {
if (parent != null) {
//2.没有则调用父类加载器的loadClass()方法;
c = parent.loadClass(name, false);
} else {
//3.若父类加载器为空,则默认使用启动类加载器作为父加载器;
c = findBootstrapClass0(name);
}
} catch (ClassNotFoundException e) {
//4.若父类加载失败,抛出ClassNotFoundException 异常后再调用自己的findClass() 方法。
c = findClass(name);
}
}
if (resolve) {
// 链接指定的类
resolveClass(c);
}
return c;
}