类加载器的作用
将class文件字节码内容加载到内存中,并将这些讲台数据转换成方法区中的运行时数据结构,在堆中生产一个代表这个类的
java.lang.Class
对象,作为方法区类数据的访问入口。
类缓存
- 标准的java se类加载器可以按要求查找类,但一旦某个类被加载到类加载器中,它将维持加载(缓存)一段时间,不过,JVM垃圾收集器可以回收这些Class对象。
类加载器的层次结构(树状结构)
1. 引导类加载器(bootstrap class loader)
- 它用来加载java的核心库(
JAVA_HOME/jre/lib/rt.jar
,或sun.boot.class.path
路径下的内容),是用原生代码来实现的,并不继承自java.lang.ClassLoader
.- 加载扩展类和引用程序加载器。并制定他们的父类加载器。
2. 扩展类加载器(extensions class loader)
- 用来加载java的扩展库(
JAVA_HOME/jre/ext/*.jar
,或java.ext.dirs
路径下的内容)。java虚拟机的实现会提供一个扩展库目录。该类加载器在此目录里面查找并加载java类。- 由
sun.misc.Launcher$ExtClassLoader
实现
3. 应用程序类加载器(application class loader)
- 它根据java应用的类路径(classpath,java.class.path路径类)一般来说,java应用的类都是由它来完成加载的。
- 由
sun.misc.Launcher$AppClassLoader
实现
4. 自定义类加载器
- 开发人员可以通过继承
java.lang.ClassLoader
类的方式实现自己的类加载器,以满足一些特殊的需求。
java.class.ClassLoader类介绍
作用:
- java.lang.Classloader类的基本职责就是根据一个指定的类的名称,找到或者生产其对应的字节代码,然后从这些字节diamante定义出一个java类,即
java.lang.Class
类的一个实例。- 除此之外,ClassLoader还负责加载java应用所需的资源,如图像文件和配置文件等。
相关方法:
- getParent() 返回该类加载器的类加载器
loadClass(String name) 加载名称为name的类,返回的记过是java.lang.Class
类的实例。
1.此方法负责加载指定名字的类,首先会从已加载的类中去寻找,如果没有找到;从parent ClassLoader[ExtClassLoader]中加载;如果没有加载到,则从Bootstrap classLoader 中尝试加载(findBootstrapClasOrNull方法),如果还是加载失败,则自己加载。如果还不能加载,则抛出异常ClassNotFoundException.
2.如果要改变类的加载顺序可以覆盖此方法;
findClass(String name) 查找名称为name的类,返回结果是java.lang.Class
类的实例。
findLoaderdClass(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$Inner
等表示方式。
类加载器的代理模式
代理模式
- 交给其他加载器来加载指定的类。
双亲委托机制
-
就是某个特定的类加载器在接到加载类的请求时,首先将加载任务委托给父类加载器,以此追溯,知道最高的爷爷辈的,如果父类加载器可以完成类加载任务,就成功返回;只有父类加载器无法完成此加载任务时,才自己去加载。
-
双亲委托机制是为了保证java核心库的类型安全。
这种机制就保证不会出现用户自己能定义java.lang.Object
类的情况 -
类加载器除了用于加载类,也是安全的最基本的屏障。
双亲委托机制是代理模式的一种
- 并不是所有的类加载器都采用双亲委托机制。
- Tomcat服务器类加载器也使用代理模式,所不同的是它是首先尝试去加载某个类,如果找不到再代理给父类加载器。这与一般类加载器的顺序是相反的。
网友评论