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

类加载过程


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

加载

三件事

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还有很多需要学习的地方

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值