类加载器学习
java指令到底干了些什么?
我们写的java代码是如何被加载到jvm内存中执行的?
类的加载过程:一个java文件从被加载到被卸载这个生命过程,总共要经历5个阶段,JVM将类加载过程分为: (加链初使卸)
加载->链接(验证+准备+解析)->初始化(使用前的准备)->使用->卸载
(1)加载
-
通过类的全限定名获取这个类的二进制字节流----->
-
将这个字节流所代表的静态存储结构转化为 方法区 的运行时数据结构----->
-
在java堆中生成一个代表这个类的Class对象,作为方法区这些数据的访问入口
(2)链接
-
验证
确保被加载类的正确性,因为类加载器会读取其它二进制流,所以为了防止恶意代码,文件格式的验证、元数据的验证、字节码验证、符号引用验证
-
准备
为类的静态变量分配内存,并将其初始化为默认值
对于基本数据类型(全局的基本类型变量没有赋值的时候可以自动赋默认值,方法中的局部变量需要在编码时手动赋默认值) 被final修饰的变量不会自动赋默认值 对于同时被static和final修饰的变量,必须在声明的时候显示的赋值 对于只被final修饰的变量,可以在声明时显示赋值,也可以在类初始化时为其赋值(例如在构造方法中) 对于引用数据类型reference(系统可以自动赋默认值null) 数组初始化时(系统可以自动赋默认值)
-
解析
把类中的符号引用转换为直接引用
-
类或接口的解析
-
数组类型还是普通对象
-
-
字段解析
-
类方法解析(先搜索父类,再搜索接口)
-
接口方法解析(先搜索父接口)
-
(3)初始化
初始化阶段是执行类的构造方法的过程
-
创建类的实例,也就是new一个对象
-
访问某个类或接口的静态变量,或者对该静态变量赋值
-
调用类的静态方法
-
反射(Class.forName(“com.crx.load”))
-
初始化一个类的子类(会首先初始化子类的父类)
-
JVM启动时标明的启动类,即文件名和类名相同的那个类
类的初始化顺序
1)如果这个类还没有被加载和链接,那先进行加载和链接
2)假如这个类存在直接父类,并且这个类还没有被初始化(注意:在一个类加载器中,类只能初始化一次),那就初始化直接的父类(不适用于接口)
3)加入类中存在初始化语句(如static变量和static块),那就依次执行这些初始化语句。
4)总的来说,初始化顺序依次是:(静态变量、静态初始化块)–>(变量、初始化块)–> 构造器;如果有父类,则顺序是:父类static方法 –> 子类static方法 –> 父类构造方法- -> 子类构造方法
类加载的底层
双亲委派机制
作用:用来挑选出一个能够去执行 加载-验证-准备-解析-初始化 的类加载器
引导类加载器:Bootstrap ClassLoader
引导类加载器:负责加载jre/lib目录下的核心类库,比如rt.jar、charsets.jar等
扩展类加载器:Extension ClassLoader
负责加载jre/ib/ext目录中的JAR类包
应用类加载器:Application ClassLoader
应用程序类加载器:负责加载ClassPath路径(类路径)下的class字节码文件,主要就是加载你自己写的那些类