JVM类加载机制

1.类加载过程

java文件通过工具编译成可以使用JVM运行的jar包,最终由java命令运行某个主类的main函数启动程序,其中,需要通过类加载器将主类加载到JVM中,主类在使用过程中,如果用到其他类,就会逐步加载这些类,
注意:JVM是不会一次性加载所有类的,只有用到该类的时候,才回去加载
类的加载过程总共有以下几步:
加载 >> 验证 >> 准备 >> 解析 >> 初始化 >> 使用 >> 卸载

  1. 加载:在硬盘上查找并通过磁盘IO读入class文件,当类使用时,JVM进行加载
  2. 验证: 校验字节码文件的正确性
  3. 准备: 给类的静态变量等分配内存,并赋予默认值
  4. 解析: 将符号引用替换为直接引用
  5. 初始化: 对垒的金泰变量初始化为程序内指定的值,并执行静态代码块

2.类加载器和双亲委派机制

上面的类加载过程主要是通过类加载器来实现,JAVA里有好几种类加载器

  1. 启动类加载器:负责加载支撑JVM运行的位于JRE的lib目录下的核心类库
  2. 扩展类加载器:负责加载支撑JVM运行的位于JRE的ext扩展目录下的jar包
  3. 应用程序类加载器: 负责加载ClassPath路径下的类包,主要加载的是你自己写的程序类
  4. 自定义加载器: 负责加载用户自定义路径下的jar包
    看一个类加载器的示例
public class TestJDKClassLoader { 
 public static void main(String[] args){ 
 System.out.println(String.class.getClassLoader()); 
 System.out.println(com.sun.crypto.provider.DESKeyFactory.class.getClassL oader().getClass().getName());
System.out.println(TestJDKClassLoader.class.getClassLoader().getClass().get Name()); 
System.out.println(ClassLoader.getSystemClassLoader().getClass().getName()); 
} 
 } 
 运行结果: 
  null 
   sun.misc.Launcher$ExtClassLoader 
   sun.misc.Launcher$AppClassLoader 
    sun.misc.Launcher$AppClassLoader

自定义一个类加载器示例
自定义加载器只需要继承ClassLoader类,实现了双亲委派机制,该类有2个核心方法

  • loadClass(String boolean)
    1). 首先,检查一下指定名称的类是否加载过,如果加载过了,就不需要重新加载,直接返回
    2)如果此类没有加载过,那么,在判断一下是否有父加载器;如果有父加载器, 则由父加载器加载,或是调用bootstrap类来加载
    3)如果父加载器及bootstrap类加载器都没有找到指定类,那么调用当前类加载器的findClass方法来完成类加载
  • findClass
    默认实现是抛出异常。所以自定义加载器需要重写findClass方法

双亲委派机制
类加载其实就有一个双亲委派机制,加载某个类时会先委托父加载器寻找目标类,找不 到再委托上层父加载器加载,如果所有父加载器在自己的加载类路径下都找不到目标类,则 在自己的类加载路径中查找并载入目标类
双亲委派机制说简单点就是,先找父亲加载,不行再由儿子自己加载

为什么要设计双亲委派机制?

  • 沙箱安全机制:自己写的java.lang.String.class类不会被加载,这样便可以防止 核心API库被随意篡改
  • 避免类的重复加载:当父亲已经加载了该类时,就没有必要子ClassLoader再加 载一次,保证被加载类的唯一性
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值