类加载机制

类是在运行时期第一次使用时动态加载的,而不是编译时期一次性加载。因为如果编译时期一次性加载,那么会占用很多的内存

1. 类的加载过程

1.1 加载

  • 通过一个类的全限定名获取该类的二进制流
  • 将该二进制流中的静态存储结构转换为方法区运行时数据结构
  • 在内存中生成该类的Class对象,作为该类的数据访问入口
  • 如果此类的父类没有加载,则先加载父类

1.2 连接

  1. 验证:验证类是否符合jvm规范、合法性、安全性检查
    1. 文件格式验证:查看字节流是否符合Class文件的规范
    2. 元数据验证:对字节码描述的信息进行语义分析,如这个类是否有父类,是否集成了不被继承的类
    3. 字节码验证:确定程序语义是否正确
  2. 准备:为static变量分配空间,设置默认值
  3. 解析:将常量池的符号引用解析为直接引用,解析动作可能在初始化之后进行

1.3 初始化

  1. 将静态代码块、static修饰的变量赋值、static final修饰的引用类型变量赋值合并成为一个<cinit>方法,在初始化时被调用。
  2. static final修饰的基本类型变量复制,在链接阶段就已经完成
  3. 初始化也是懒惰执行的
  4. 子类初始化之前会初始化父类
  5. 初始化时懒惰执行的。

2. 触发类的初始化情况

2.1 主动引用一定触发类的初始化

  • new一个类的对象
  • 调用类的静态方法和静态成员
  • 反射生成一个类对象会触发其类初始化
  • 子类初始化会先进行父类初始化

2.2 被动引用不会触发类的初始化

  • 当访问一个静态域,只有声明这个域的类会初始化。(如通过子类引用调用父类静态变量,只会导致父类初始化,而不会导致子类初始化)
  • 引用常量不会触发初始化(final修饰的且在定义时声明初始值的也算常量)
  • 通过数组定义类引用,不会触发此类的初始化。

3. 双亲委派机制

3.1 类加载器的优先级

  • 启动类加载器:加载JAVA_HOME/jre/lib下的jar包
  • 扩展类加载器:加载JAVA_HOME/jre/lib/ext
  • 应用程序类加载器:加载classpath路径下
  • 自定义加载器

3.2 什么是双亲委派机制

  • 在加载类的时候,优先委派给上级类加载器进行加载。对于上级类加载器
    • 如果能找到这个类,则由上级加载,加载后该类也对下级加载器可见
    • 如果找不到这个类,则下级类加载器才有资格执行加载
  • 总结:类的加载首先请求父类加载器加载,父类加载器无能为力时才由子类加载器自行加载

3.3 双亲委派机制的目的

  • 让上级类加载器中的类对下级共享。以便我们自己写的类能够依赖于jdk提供的核心类
  • 让类的加载有优先次序,保证jdk提供的核心类优先加载。(而且只能由特定类加载器去加载)

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值