类装载器ClassLoader工作机制

类装载器ClassLoader
类装载器工作机制
类装载器就是寻找类的节码文件并构造出类在JVM 内部表示对象的组件。在Java 中,
类装载器把一个类装入JVM 中,要经过以下步骤:
1.装载:查找和导入Class 文件;
2.链接:执行校验、准备和解析步骤,其中解析步骤是可以选择的:
a)校验:检查载入Class 文件数据的正确性;
b)准备:给类的静态变量分配存储空间;
c)解析:将符号引用转成直接引用;
3.初始化:对类的静态变量、静态代码块执行初始化工作。
类装载工作由ClassLoader 及其子类负责,ClassLoader 是一个重要的Java 运行时系统
组件,它负责在运行时查找和装入Class 字节码文件。JVM 在运行时会产生三个
ClassLoader:根装载器、ExtClassLoader(扩展类装载器)和AppClassLoader(系统类装载
器)。其中,根装载器不是ClassLoader 的子类,它使用C++编写,因此我们在Java 中看不
到它,根装载器负责装载JRE 的核心类库,如JRE 目标下的rt.jar、charsets.jar 等。
ExtClassLoader 和AppClassLoader 都是ClassLoader 的子类。其中ExtClassLoader 负责装载
JRE 扩展目录ext 中的JAR 类包;AppClassLoader 负责装载Classpath 路径下的类包。
这三个类装载器之间存在父子层级关系,即根装载器是ExtClassLoader 的父装载器,
ExtClassLoader 是AppClassLoader 的父装载器。默认情况下,使用AppClassLoader 装载应
用程序的类,我们可以做一个实验:
代码清单 3-11 ClassLoaderTest
public class ClassLoaderTest {
public static void main(String[] args) {
ClassLoader loader = Thread.currentThread().getContextClassLoader();
System.out.println("current loader:"+loader);
System.out.println("parent loader:"+loader.getParent());
System.out.println("grandparent loader:"+loader.getParent(). getParent());
}
}
运行以上代码,在控制台上将打出以下信息:
current loader:sun.misc.Launcher$AppClassLoader@131f71a
parent loader:sun.misc.Launcher$ExtClassLoader@15601ea
//①根装载器在Java中访问不到,所以返回null
grandparent loader:null
通过以上的输出信息,我们知道当前的ClassLoader 是AppClassLoader,父ClassLoader
是ExtClassLoader,祖父ClassLoader 是根类装载器,因为在Java 中无法获得它的句柄,
所以仅返回null。
JVM 装载类时使用“全盘负责委托机制”,“全盘负责”是指当一个ClassLoader 装载
一个类的时,除非显式地使用另一个ClassLoader,该类所依赖及引用的类也由这个
ClassLoader 载入;“委托机制”是指先委托父装载器寻找目标类,只有在找不到的情况下
才从自己的类路径中查找并装载目标类。这一点是从安全角度考虑的,试想如果有人编写
了一个恶意的基础类(如java.lang.String)并装载到JVM 中将会引起多么可怕的后果。但
是由于有了“全盘负责委托机制”,java.lang.String 永远是由根装载器来装载的,这样就避
免了上述事件的发生。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值