类加载器
当程序运行以后,第一次使用某个类的时候,会将这个类的class文件读取到内存,并将此类的所有信息存储到一个Class对象中。
类的加载时机
- 创建类的实例
- 类的静态变量或为静态变量赋值
- 类的静态方法
- 使用反射方式来强制创建某个类或者接口对应的java.lang.Class对象。
- 初始化某个类的子类
- . 直接使用java.exe命令来运行某个主类。
以上六种情况的任何一种,都可以导致JVM将一个类加载到方法区。
类加载器的种类:
类加载器:是负责将磁盘上的某个class文件读取到内存并生成Class的对象。
Java中有三种类加载器,它们分别用于加载不同种类的类
class: 启动类加载器(Bootstrap ClassLoader):用于加载系统类库<JAVA_HOME>\bin目录下的 class。
扩展类加载器(Extension ClassLoader):用于加载扩展类库<JAVA_HOME>\lib\ext目录下的 class。
应用程序类加载器(Application ClassLoader):用于加载我们自定义类的加载器。
双亲委派机制
上图展示了双亲委派机制的层次关系
“双亲委派模型"中,除了顶层的启动类加载器外,其余的类加载器都有自己的"父级类加载器”。
某个"类加载器"收到类加载的请求,它首先不会尝试自己去加载这个类,而是把请求交给父级类加载器。
因此,所有的类加载的请求最终都会传送到顶层的"启动类加载器"中。如果"父级类加载器"无法加载这个类,然后子级类加载器再去加载。
双亲委派机制的一个显而易见的好处是:Java的类随着它的类加载器一起具备了一种带有优先级的层次关系。例如:java.lang.Object。它存放在rt.jar中。无论哪一个类加载器要加载这个类,最终都是委派给处于顶端的"启动类加载器"进行加载,因此java.lang.Object类在程序的各种类加载器环境中都是同一个类。
相反,如果没有"双亲委派机制",那么用户自己写了一个java.lang.Object,那么当我们编写其它类时,这种隐式的继承使用的将会是用户自己编写的java.lang.Object类,那么这个就会变得十分的混乱。