JVM类加载机制

一、类加载步骤

 

 JVM采用的是懒加载机制,即只有类在被使用时才加载。类的加载主要可分为以下几步:

加载:即把类字节码文件加载到内存,在一步会在内存中生成一个代表这个类的java.lang.Class对象,作为方法区这个类的各种数据的访问入口。

验证:校验字节码文件的准确性,每个字节码文件都有固定的格式,以此来校验字节码文件是否被损坏。

准备:给静态变量分配内存空间并赋与默认值。

解析:符号引用替换为直接引用,该阶段会把一些静态方法(符号引用,比如 main()方法)替换为指向数据所在内存的指针或句柄等(直接引用),这是所谓的静态链接过程(类加载期间完成),动态链接是在程序运行期间完成的将符号引用替换为直接引用。

初始化:对类变量初始化赋值,并执行静态代码块。

二、类加载器

 JVM的类加载主要有四类加载器来配合完成。

引导类加载器(BootstrapClassLoad):负责加载支撑JVM运行的位于JRE的lib目录下的核心类库,比如 rt.jar、charsets.jar等。为避免核心类库被篡改,这类核心类库只能有引导类加载器加载,即使自定义类加载指定去加载该路径下的类也会报错无法加载成功。引导类加载器由C++代码实现,因此,在JAVA程序中对它的引用值为null。
扩展类加载器(ExtClassLoad):负责加载支撑JVM运行的位于JRE的lib目录下的ext扩展目录中的JAR类包。
应用程序类加载器(AppClassLoad):负责加载ClassPath路径下的类包,主要就是加载你自己写的那些类。
自定义加载器:负责加载用户自定义路径下的类包。自定义加载器需要继承ClassLoad类,同时,可根据自身业务需要是否要重新其加载类的loadClass和findClass方法。
类加载器从上往下构成"父子关系",注意,是在加载器类中有一个parent属性的值为上一层类加载器对象,并非JAVA层面的类继承关系。

三、类加载机制

JVM的类加载机制采用的是双亲委派机制,当加载一个类时,如果使用自定义类加载器,则从自定义加载器查看该加载器是否加载过此类,如果没有则委托上层的应用程序类加载器去加载,在加载时,同样的检查当前类加载器是否加载过此类,如果没有则委托上层的扩展类加载器去加载。在加载时,同样的检查当前类加载器是否加载过此类,没有则委托上层的引导类加载器去加载,在加载时,同样的检查当前类加载器是否加载过此类,如果没有,则由引导类加载器去加载此类,如果在引导类加载器管辖的路径下没加载到此类,则有下层的扩展类程序类加载器去加载,如果在扩展类加载器管辖的路径下没加载到此类,则有下层的应用程序类加载器去加载;如果在应用程序类加载器管辖的路径下没加载到此类,则有下层的自定义类加载器去加载;再加载不到则抛出异常。可以总结成一句话:从下往上检查类是否被加载过,从上往下去加载类。

 四、双亲委派机制的意义

    沙箱安全机制:各个类加载器管辖自己路径下类的加载,比如核心类库只能有引导类加载器加载,可以防止核心类库被篡改。自己写一个跟核心类库一模一样类名、包名的Object、String类,是不会被加载成功的。
    避免类的重复加载:当父亲已经加载了该类时,就没有必要子ClassLoader再加载一 次,保证被加载类的唯一性。
    提升类加载效率:因为在应用程序中,90%以上的类都是程序员自己写的类,所以从下往上开始检查能提升类加载的效率。

五、全盘负责委托机制

全盘负责 ”是指当一个 ClassLoder 装载一个类时,除非显示的使用另外一个 ClassLoder ,该类
所依赖及引用的类也由这个 ClassLoder作为入口 载入,由它来触发类加载。

六、Tomcat打破双亲委派机制

   tomcat为什么不用双亲委派机制?

    一个tomcat可能运行多个应用程序,而且这个多个应用程序引用了同一个类的不同版本,所以,从双亲委派机制保证类只加载一份的特点不符合tomcat的业务场景需求。

   tomcat如何保证jps的热加载?   

   每个jsp文件对应一个唯一的类加载器,当一个jsp文件修改了,就直接卸载这个jsp类加载器。重新创建类加载器,重新加载jsp文件
   tomcat的几个主要类加载器:
commonLoader:Tomcat最基本的类加载器,加载路径中的class可以被Tomcat容器本身以及各个Webapp访问;
catalinaLoader:Tomcat容器私有的类加载器,加载路径中的class对于Webapp不 可见;
sharedLoader:各个Webapp共享的类加载器,加载路径中的class对于所有Webapp可见,但是对于Tomcat容器不可见;
WebappClassLoader:各个Webapp私有的类加载器,加载路径中的class只对当前 Webapp可见,比如加载war包里相关的类,每个war包应用都有自己的WebappClassLoader,实现相互隔离,比如不同war包应用引入了不同的spring版本, 这样实现就能加载各自的spring版本;

注意: 同一个JVM内,两个相同包名和类名的类对象可以共存,因为他们的类加载器可以不一样(同一个类加载器Class,两个类加载器实例也称为不一样),所以看两个类对象是否是同一个,除了看类的包名和类名是否都相同之外,还需要他们的类加载器对象也是同一个才能认为他们是同一个。

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

季风泯灭的季节

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值