1、类的加载
定义:将类的.class文件中二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在堆区创建一个
java.lang.Class对象, 用来封装在方法区内的数据结构;
类加载最终产品为位于堆区中class对象,class对象封装了类在方法区中的数据结构,并向Java程序员提供
了访问方法区内数据结构的接口;
注意:类加载器并不是要等到某个类被 "首次主动使用" 时再加载它,JVM规范允许类加载器预先加载即将要被使用的类,
如果预先加载过程中存在 .class文件缺失或存在错误,类加载器必须在程序首次使用该类时才报告错误(LinkageError),
如果这个类一直没有被程序主动使用,那么类加载器就不会报错;
2、类的生命周期
加载:通过一个类的全限定名来获取其定义的二进制字节流;
将这个字节流所代表的静态存储结构转化为方法区的运行时数据结构;
在Java堆中生成一个代表这个类的java.lang.Class对象,作为对方法区中数据的访问入口;
验证:确保被加载的类的正确性,包含文件格式验证、元数据验证、字节码验证、符号引用验证;
准备:为类的静态变量分配内存,并将其初始化为默认值;
(1)准备阶段进行内存分配仅包括类变量(static);
(2)所设置的初始值通常情况下是数据类型默认的零值(如0、0L、null、false等);
public static int value = 3;
value变量在准备阶段过后的初始值是0,而不是3;把value赋值为3的动作在初始化阶段才会执行;
(3)如果类变量同时被static 、final修饰,那么在准备阶段变量就会被初始化成其显式指定的值;
public static final int value = 3 ;
value变量在准备阶段过后的初始值是3;
解析:把类中的符号引用转换为直接引用;
初始化:为类的静态变量赋予正确的初始值;
3、类加载器
几种类加载器层次图:
JVM角度分析类加载器:
启动类加载器,JVM自身一部分;
其他类加载器,独立于JVM,全部继承自抽象类java.lang.ClassLoader,这些类加载器需要由启动类加载器加载到
内存中之后才能加载其他的类;
Java开发人员角度分析类加载器:
启动类加载器,负责加载存放在JDK\jre\lib下 或 被Xbootclasspath参数指定的路径中,并且能被jvm识别类库;
启动类加载器是无法被java程序直接引用的;
扩展类加载器,负责加载JDK\jre\lib\ext下 或 java.ext.dirs系统变量指定的路径中的所有类库 ;
开发者可以直接使用扩展类加载器;
应用程序类加载器,负责加载用户类路径所指定的类,开发者可以直接使用该类加载器;
4、jvm类加载机制
全盘负责:当一个类加载器负责加载某个class时,该class所依赖的和引用的其他class也将由该加载器负责加载;
父类委托:先让父类加载器试图加载该类,如果父类加载器无法加载该类时才尝试从自己的类路径加载中加载该类;
缓存机制:缓存机制保证所有被类加载器加载过class都会被缓存;当程序中需要使用某个class时,类加载器先从缓存
区寻找该class,如果缓存区不存在,类加载器就会去读取该类对应的二进制数据,并将其转换成class对象,
存入缓存区;所以,修改class文件后,必须重启JVM,修改后的程序才会生效;
5、类加载方式
三种方式: 命令行启动应用程序时由JVM初始化加载;
通过Class.forName()方法动态加载;
通过ClassLoader.loadClass()方法动态加载;
例子:
6、双亲委派模型
定义:如果一个类加载器接收到了类加载的请求,它首先不会尝试自己去加载这个类,会委托给其父类加载器,依次向上,
直到到达最顶级的启动类加载器(BootstrapClassLoader);如果当父类加载器无法加载该类时,子类加载器才会尝试
去加载该类;
举例:(1) 当AppClassLoader加载一个class时,它不会尝试去加载此Class,而是将加载请求委托给父类加载
器ExtClassLoader;
(2)当ExtClassLoader加载一个class时,它不会尝试去加载此Class,而是将加载请求委托给父类加载
器BootStrapClassLoader;
(3)当BootStrapClassLoader加载失败(在JDK/jre/lib里未查找到该class),会使用ExtClassLoader去加载;
(4)当ExtClassLoader加载失败,会使用AppClassLoader去加载,如果AppClassLoader加载失败,则会
报出异常ClassNotFoundException;
7、参考
http://www.ityouknow.com/jvm/2017/08/19/class-loading-principle.html