java自带三种加载器
bootstrap
启动类加载器
默认加载jdk中的rt.jar包中的class文件,比如Object 、Array、String等等,这就是我们安装jdk后直接能使用这些类的原因,启动类加载器使用的是C++语音写的
测试:
@Test
public void classLoaderTest() {
Object object = new Object();
System.out.println(object.getClass().getClassLoader());
System.out.println(object.getClass().getClassLoader().getParent());
}
因为启动类加载器是C++语音些java识别不了所以运行结果是null
extensions
扩展类加载器
加载JAVA_HOME/lib/ext目录下或者由系统变量-Djava.ext.dir指定位路径中的类库,开发者可以直接使用标准扩展类加载器。
application
应用类加载器
Sun公司实现的sun.misc.Launcher$AppClassLoade,也就是我们开发这自己写的类一般用到的加载器
测试:
public void classLoaderTest() {
Object object = new Object();
System.out.println(object.getClass().getClassLoader());
MyObject myObject = new MyObject();
System.out.println(myObject.getClass().getClassLoader());
System.out.println(myObject.getClass().getClassLoader().getParent());
System.out.println(myObject.getClass().getClassLoader().getParent().getParent());
}
这里也可以看出来 应用类加载器父类是扩展类加载器,扩展类加载器的父类是启动类加载器
双亲委派
当一个类收到一个类加载请求,他不会自己尝试加载这个类,而是把这个请求委派给父类,每个层次的类加载器都是如此,只有父类加载器无法完成加载时候,子类加载器才会尝试自己去加载.这就保证了java.lang.Object无论是那个类去加载最后都会得到同一个Object对象
package java.lang;
/**
* @author: ZZP
* @create: 2020-06-10 15:00
**/
public class Object {
public static void main(String[] args) {
System.out.println("test");
}
}
这是我自己写一个类对象Object
运行结果
错误: 在类 java.lang.Object 中找不到 main 方法, 请将 main 方法定义为:
public static void main(String[] args)
否则 JavaFX 应用程序类必须扩展javafx.application.Application
类加载器会从bootstrap加载器中开始,这样也保证了沙箱安全也就是保证了jdk中的代码永远是最先调用的