JVM核心机制_深入类加载器_层次结构(三种类加载器)_代理加载模式_双亲委派机制

java.class.ClassLoader类

• 作用:

– java.lang.ClassLoader类的基本职责就是根据一个指定的类的名称, 找到或者生成其对应的字节代码,然后从这些字节代码中定义出一个 Java 类,即 java.lang.Class类的一个实例。 – 除此之外,ClassLoader还负责加载 Java 应用所需的资源,如图像文 件和配置文件等。

• 相关方法

 – getParent()           返回该类加载器的父类加载器。
 loadClass(String name)      加载名称为 name的类,返回的结果是 java.lang.Class类的实例。
 findClass(String name)      查找名称为 name的类,返回的结果是 java.lang.Class类的实例。
 findLoadedClass(String name)   查找名称为 name的已经被加载过的类,返回的结果是 java.lang.Class类的实 例。 
 defineClass(String name, byte[] b, int off, int len)   把字节数组 b中的内容转换成 Java 类,返回的结果是 java.lang.Class类的实例。这个方法被声明为 final的。 resolveClass(Class<?> c)     链接指定的 Java 类。 
   – 对于以上给出的方法,表示类名称的 name参数的值是类的二进制名称。需要注意的是内部类的表示,如 com.example.Sample$1和com.example.Sample$1Inner等表示方式。 

类加载器的层次结构(树状结构)

• 引导类加载器(bootstrap class loader)

– 它用来加载 Java 的核心库(JAVA_HOME/jre/lib/rt.jar,或sun.boot.class.path路径下的 内容),是用原生代码来实现的,并不继承自 java.lang.ClassLoader。
– 加载扩展类和应用程序类加载器。并指定他们的父类加载器。

• 扩展类加载器(extensions class loader)

– 用来加载 Java 的扩展库(JAVA_HOME/jre/ext/*.jar,或java.ext.dirs路径下的内容) 。
 Java 虚拟机的实现会提供一个扩展库目录。该类加载器在此目录里面查找并加载 Java 类。 
– 由sun.misc.Launcher$ExtClassLoader实现

•应用程序类加载器(application class loader)

  – 它根据 Java 应用的类路径(classpath, java.class.path 路径下的内容)来加载 Java 类。 
  一般来说,Java 应用的类都是由它来完成加载的。
   – 由sun.misc.Launcher$AppClassLoader实现 

• 自定义类加载器

   – 开发人员可以通过继承 java.lang.ClassLoader类的方式 实现自己的类加载器,以满足一些特殊的需求。

加载器之间的关系

在这里插入图片描述

类加载器的代理模式

• 代理模式
– 交给其他加载器来加载指定的类
• 双亲委托机制
– 就是某个特定的类加载器在接到加载类的请求时,首先将加载任务委 托给父类加载器,依次追溯,直到最高的爷爷辈的,如果父类加载器 可以完成类加载任务,就成功返回;只有父类加载器无法完成此加载 任务时,才自己去加载。
– 双亲委托机制是为了保证 Java 核心库的类型安全。
• 这种机制就保证不会出现用户自己能定义java.lang.Object类的情况。
– 类加载器除了用于加载类,也是安全的最基本的屏障。
• 双亲委托机制是代理模式的一种
– 并不是所有的类加载器都采用双亲委托机制。
– tomcat服务器类加载器也使用代理模式,所不同的是它是首先尝试去加载某个类,如果找不到再代理给父类加载器。 这与一般类加载器的顺序是相反的

个人理解:

代理模式:
  就是假如一个自定义加载器要加载A类,它不加载直接给它的父类加载器应用程序加载类(application class loader),它也会继续给它的父类 扩展类加载器(extensions class loader) ,它同样也会继续上传到自己的父类引导类加载器(bootstrap class loader),

到了引导类加载器时它会加载A类,如果加载不了,会传递到扩展类加载器(extensions class loader) ,如果它也加载不了会继续给下一级,直到最后自定义加载器时如果还不能加载则报错。

双亲委托机制:

安全性安全在,假如我自定义一个Java.lang.String类,里面重写toString方法,return “aa”;
在另一个类中写

String a="lzy";
	System.out.println(a.toString());

运行结果是:lzy
而不是aa
因为采用双亲委托机制,加载一个类从最高类加载器–引导类加载器(bootstrap class loader),这里面有真正的toString类,所以就会用这个,而不是你自己写的类。

java.class.ClassLoader类API

• 相关方法
– getParent() 返回该类加载器的父类加载器。
– loadClass(String name) 加载名称为 name的类,返回的结果是 java.lang.Class类的实例。
• 此方法负责加载指定名字的类,首先会从已加载的类中去寻找,如果没有找到;从parent ClassLoader[ExtClassLoader]中加载;如 果没有加载到,则从Bootstrap ClassLoader中尝试加载(findBootstrapClassOrNull方法), 如果还是加载失败,则自己加载。如果还 不能加载,则抛出异常ClassNotFoundException。
• 如果要改变类的加载顺序可以覆盖此方法;
– findClass(String name) 查找名称为 name的类,返回的结果是 java.lang.Class类的实例。
– findLoadedClass(String name) 查找名称为 name的已经被加载过的类,返回的结果是 java.lang.Class类的实 例。
– defineClass(String name, byte[] b, int off, int len) 把字节数组 b中的内容转换成 Java 类,返回的结果是 java.lang.Class类的实例。这个方法被声明为 final的。
– resolveClass(Class<?> c) 链接指定的 Java 类。 –
–表示类名称的 name参数的值是类的名称。需要注意的是内部类的表示,如 com.example.Sample 1 和 c o m . e x a m p l e . S a m p l e 1和 com.example.Sample 1com.example.SampleInner等表示方式。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值