类加载器的分类
-
JVM支持两种类型的类加载器,分别为引导类加载器(BootStrap ClassLoader)和自定义类加载器(User-Defined ClassLoader)
-
从概念上讲,自定义类加载器一般指的是程序中由开发人员自定义的一类类加载器,但是java虚拟机规范却没有这么定义,而是将所有派生于抽象类ClassLoader的类加载器都划分为自定义类加载器
-
无论类加载的类型如何划分,在程序中我们最常见的类加载器始终只有3个,即:
启动类加载器(引导类加载器,BootStrap ClassLoader)
- 这个类加载器是由c/c++语言实现的,嵌套在JVM内部
- 它用来加载Java的核心库(JAVA_HOME/jre/lib/rt.jar,resources.jar或sun.boot.class.path路径下的内容),用于提供JVM自身需要的类
- 并不继承自java.lang.ClassLoader,没有父加载器
- 加载扩展类和应用程序类加载器,并指定为他们的父类加载器
- 出于安全考虑,BootStrap启动类加载器只加载包名为java,javax,sun等开头的类
- 当获取 一个Class的类加载器时,返回null,说明它是由引导类加载器加载的,所以扩展类加载器getParent返回null
扩展类加载器(Extension ClassLoader)
- java语言编写,由sun.misc.Launcher$ExtClassLoader实现
- 派生于ClassLoader类
- 父类加载器为启动类加载器
- 从java.ext.dirs系统属性所指定的目录中加载类库,或从JDK的安装目录的jre/lib/ext子目录(扩展目录)下加载类库.如果用户创建的jar放在此目录下,也会自动由扩展类加载器加载
应用程序类加载器(系统类加载器 AppClassLoader)
- java语言编写,由sun.misc.Launcher$AppClassLoader实现
- 派生于ClassLoader类
- 父类加载器为扩展类加载器
- 它负责加载环境变量classpath或系统属性java.class.path指定路径下的类库
- 该类加载器是程序中默认的类加载器,一般来说,java应用都是由它来完成加载的
- 通过ClassLoader.getSystemClassLoader()来获取该类加载器
用户自定义类加载器
- 在java的日常应用程序开发中,类的加载几乎是由上述3种类加载器相互配合执行的,在必要时,我们还可以自定义类加载器,来定制类的加载方式
- 为什么要自定义加载器
- 隔离加载器
- 修改类加载的方式
- 扩展加载源
- 防止源码泄露
- 自定义类加载器实现步骤
- 继承抽象类java.lang.ClassLoader
- 在jdk1.2之前,在自定义类加载器时,总会去继承ClassLoader类并重写loadClass()方法,从而实现自定义的类加载器,但在jdk1.2之后已不再建议用户去覆盖loadClass()方法,而是建议把自定义的类加载逻辑编写再findClass()方法中
oadClass()方法,而是建议把自定义的类加载逻辑编写再findClass()方法中 - 在编写自定义类加载器时,如果没有太过于复杂的需求,可以直接继承URLClassLoader类,这样可以避免自己去编写findClass()方法及其获取字节码流的方式,使自定义类加载器编写更加简洁