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