一、类加载器(ClassLoader)负责加载class文件,class文件在文件开头有特定的文件标识,而且ClassLoader只负责 class 文件的加载,至于class文件是否可以运行则由Execution Engine决定;java
二、类的加载过程安全
加载class文件 ---> 链接(验证、准备、解析class文件)----> 初始化类的对象 ----> 使用类的对象 -----> 卸载。jvm
(1)加载class文件spa
取类的二进制字节流,经过类的全量命名,将静态存储结构房到方法区中,Class(类的定义或结构,即类对象)放到堆中; 设计
(2)初始化类对象3d
执行类的构造器<clinit>,为类的静态变量赋予正确的初值;code
构造器包含:static变量、static块;根据代码中的顺序来决定先执行哪一个;对象
先执行构造器(static变量、static块),再执行构造方法;blog
三、JDK中的类加载器:递归
<1> 启动类加载器(BootStrap ClassLoader): C++编写的,虚拟机自带的加载器;加载$JAVA_HOME/jre/lib/rt.jar
<2> 扩展类加载器(Extension ClassLoader):java编写;加载$JAVA_HOME/jre/lib/ext/*.jar
<3> 应用程序类加载器(App ClassLoader):java编写;也叫系统类加载器,加载当前应用的 $classpath 下全部类;
<4> 用户自定义加载器:Java.Lang.ClassLoader 的子类,用户能够定制类的加载方式;
三、类加载器说明
第一步:将 MyHello 类打包成 jar包,而后放入$JAVA_HOME/jre/lib/ext/ 目录下。 MyHello内容以下:
public class MyHello { public static void main(String[] args) { } }
第二步:编写以下代码
public class Demo1 { public static void main(String[] args) throws ClassNotFoundException { //启动类加载器(BootStrap ClassLoader)
Object object = new Object(); System.out.println("Object-classLoader: " + object.getClass().getClassLoader()); //结果: Object-classLoader: null //扩展类加载器 (Extension ClassLoader)
Class myHello = Class.forName("com.yufeng.jvm.hello.MyHello"); System.out.println("MyHello-classLoader: " + myHello.getClassLoader()); //结果:MyHello-classLoader: sun.misc.Launcher$ExtClassLoader@7ea987ac //应用类加载器 (App ClassLoader)
Demo1 classLoaderDemo1 = new Demo1(); System.out.println("Demo1-classLoader: " + classLoaderDemo1.getClass().getClassLoader()); //①结果:Demo1-classLoader: sun.misc.Launcher$AppClassLoader@18b4aac2
System.out.println("Demo1-classLoader-parent: " + classLoaderDemo1.getClass().getClassLoader().getParent()); //②结果:Demo1-classLoader-parent: sun.misc.Launcher$ExtClassLoader@2503dbd3
System.out.println("Demo1-classLoader-parent-parent: " + classLoaderDemo1.getClass().getClassLoader().getParent().getParent()); //③结果:Demo1-classLoader-parent-parent: null
} }
从以上的①、②、③能够看出,启动类加载器是扩展类加载器的父类,扩展类加载器是应用加载器的父类。
打印出的Object的类加载器为null,表示它是根加载器,即启动类加载器。
四、类加载器的双亲委派机制
某个特定的类加载器在接到加载类的请求时,首先将加载任务委托给父类加载器,依次递归,若父类加载器能够完成类加载任务,就成功返回;只有父类加载器没法完成此加载任务时,才本身去加载。
这样设计为了保证java的一种安全特性--------沙箱机制(防止恶意代码对java的破坏);
例:若本身新建了一个String类(classpath路径下),则类加载器是应用程序类加载器;加载的时候程序类加载器委托父类加载器加载(即扩展类加载器),扩展类加载器去委托启动类加载器去加载,rt.jar中有String.class文件,则会加载这个Class,不会加载本身新建的那个String类。