反射-类加载器(获取类的加载器、获得系统类加载器可以加载的路径、双亲委派机制)
作者:互联网
类加载器的作用 类加载的作用:将class文件字节码内容加载到内存中,并将这些静态数据转换成方法区的运行时 数据结构,然后在堆中生成一个代表这个类的java.lang.Class对象,作为方法区中类数据的访问入口。 类缓存∶标准的JavaSE类加载器可以按要求查找类,但一旦某个类被加载到类加载器中,它将维持加载(缓存)一段时间。不过JVM垃圾回收机制可以回收这些Class对象 执行顺序: 源程序(*.java文件)--》java编译器---》字节码(*.class文件)---》类装载器---》字节码检验器---》解释器----》操作系统平台
类加载器的类型 类加载器作用是用来把类(class)装载进内存的。JVM规范定义了如下类型的类的加载器。 最基本的:引导类加载器:用C++编写的,是JVM自带的类加载器,负责java平台核心库(rt.jar),用来装载核心类库,该加载器无法直接获取。 扩展类加载器:负责jre/lib/ext目录下的jar包或-D java。ext.dirs 指定目录下的jar包装入工作库 系统类加载器:负责java-classpath或-Djava.class.path所指的目录下的类与jar包装入工作,是最常用的加载器
//获取系统类的加载器 public class Main { public static void main(String[] args) throws ClassNotFoundException{ //获取系统类的加载器 ClassLoader systemClassLoader=ClassLoader.getSystemClassLoader(); System.out.println(systemClassLoader); //获取系统类加载器的父类加载器--》扩展加载器 ClassLoader parent= systemClassLoader.getParent(); System.out.println(parent); //获取扩展类加载器的父类加载器---》根加载器(c/c++) ClassLoader parent1=parent.getParent(); System.out.println(parent1);//因为这个加载是我们无法直接获取的,所以会返回一个null //测试当前类是那个加载器加载的 ClassLoader classLoader=Class.forName("Main").getClassLoader; System.out.println(classLoader); //测试JDK内置的类是谁加载的 classLoader1=Class.forName("java.lang.Object").getClassLoader; System.out.println(classLoader1);//因为根加载类,我们无法直接获取,所这个是根加载器加载的 //如何获得系统类加载器可以加载的路径 System.out.println(System.getProperty("java.class.path"));//获得的路径表示,我们的类只能在这些地方被加载,如果不在这些地方那我们的类,就不能被加载 //双亲委派机制 //比如我们要定义一个java.lang.String这个类,那这个机制会一个一加载的往上找,先加当前类的加载有没有这个包和类,再找内置加载器中的包有没有这个包和类,最后去根加载器中包找这个包和类,如果有这个那我们自定义的就不能用了 } }
双亲委派机制,的确是防止同名包、类与 jdk 中的相冲突,实际上加载类的时候,先通知 appLoader,
看 appLoader 是否已经缓存,没有的话,appLoader 又委派给他的父类加载器(extLoader)询问,
看他是不是能已经缓存加载,没有的话,extLoader 又委派他的父类加载器(bootstrapLoader)询问,
BootstrapLoader看是不是自己已缓存或者能加载的,有就加载,没有再返回 extLoader,extLoader
能加载就加载,不能的话再返回给 appLoader 加载,再返回的路中,谁能加载,加载的同时也加缓存里。
正是由于不停的找自己父级,所以才有 Parents 加载机制,翻译过来叫 双亲委派机制
标签:委派,java,System,双亲,println,out,class,加载 来源: https://www.cnblogs.com/huxingchen/p/16351938.html