zl程序教程

您现在的位置是:首页 >  后端

当前栏目

JVM深入学习笔记六-JVM类加载

JVM笔记学习 深入 加载
2023-09-14 08:58:00 时间
p span >类加载过程


主要分为了 加载- 链接(验证- 准备- 解析)- 初始化- 使用- 卸载这几个阶段。

加载

三件事

1. 通过类的权限定名称来获取定义此类的二进制字节流(可以是文件,网络,数据库,动态等等等等)

2. 把类的结构放在方法区中

3. 创建Class对象作为访问入口

验证

主要包括了字节码验证,元数据验证(这部分在编译期间基本上避免了),类文件格式验证。

准备

包括了内存分配和类变量(static)初始值的设定,以及常量池的写入。

解析

主要是解析符号引用和直接引用,符号引用就是个字面量,直接引用是真正的内存地址。

初始化

就是执行 clinit ()方法的过程

它是 由类变量赋值和static代码块合并成的

会先执行父类的clinit方法。

JVM会保证clinit方法的线程安全,比如写一个static的长执行方法,两个线程调用的时候会看到等待过程

初始化触发的时间点  new , 引用类的final字段,但是staticfinal的除外,  调用类的静态方法。  用reflect包反射调用,执行时main方法宿主类, 子类初始化先初始化父类


类加载器

不同的类加载器使用instanceof的时候会判断两个相同的Class生成的obj是返回false的。

双亲委派模型

每个类加载器的功能图上已经说得很清楚了。

说是双亲委派模型主要是说得1.2之后的所有的类加载器的实现的模式都是

实现的时候就是我们自己去实现findClass()方法,loadClass()方法里的逻辑是先调用了父类类加载器,当父类无法加载的时候调用findClass()方法。


破坏双亲委派模型

双亲的模型会有一个问题,默认了用户类会调用基础类,但是有些情况不是这样的,比如JNDI,JNDI是由启动加载器加载的,但是其确需要引用一些第三方的类,为了解决类似的情况,引入了ThreadContextClassLoader. JNDI, JDBC等都使用了这种情况


此外,为了热部署的OSGI技术也是一个没有使用这个结构的代表。



Tomcat类加载器架构

一个Web容器设计类加载的时候可能会遇到的问题:

1. 每个应用应该相互隔离, 因为可能用到了同一个类的不同版本

2. 各应用所使用的类库又应该可以共享,这样就不用每一个都拷贝一份servelt.jar jsp.jar等等

3. 服务器本身的类库应该与应用的类库相互隔离

4. 应该支持JSP生成类的热替换

所以tomcat的类加载器设计成这个样子的:



可以看出,出了JDK自己的加载器外,还有tomcat设计的类加载的目录

/common  可以被Tomcat和所有的Web应用共同使用的

/server   可以被Tomcat使用,不能被Web应用使用

/shared 被所有Web程序共同使用

/WebApp/WEB-INF/ 仅此Web应用可见

这样的类设计可以很好的实现上述的功能


OSGI灵活的类加载器

模块化技术的事实标准。

OSGI给每一个bundle使用了一个类加载器,当需要更换bundle的时候连同类加载器一起换掉。

比如BundleA对应了ClassLoaderA, 其通过Export到处了一个包,  这个包被BundleB所引用。

bundle自己的类加载器负责加载自己的类,以及调用启动类加载器加载rt.jar等公用内容。

而当bundleB需要加载Import A的包的时候就会委派给ClassLoaderA进行加载,因为不在同一个ClassLoader中同名类是可以共存的, 这样就保证了可以同时存在不同版本的jar包在开发环境中。

其类加载器的结构



对于OSGI还有很多需要学习的地方


JVM类加载过程 相信有一部分java程序员不是太清楚虚拟机是如何将类从java代码变成class文件,再从class文件到到内存,再将我们写的程序转化成具体的程序的,这里就总结下这个过程。
JVM类加载过程 对于数组类而言,情况就有所不同,数组类本身不通过类加载器创建,它是由Java虚拟机直接在内存中动态构造出来的。
JVM模块 | JVM的组成、内存模型、类加载 JVM是java虚拟机,由四个部分组成,分别为:ClassLoader(类加载器),Runtime Data Area(运行时数据区,内存分区),Execution Engine(执行引擎),Native Interface(本地库接口)