类加载的作用:将class文件字节码内容加载到内存中,并将这些静态数据转换成方法区的运行时数据结构,然后在堆中生成一个代表这个类的java,lang.Class对象,作为方法区中类数据的访问入口
类缓存:标准的javaSE类加载器可以按照要求查找类,但一旦某个类被加载到类加载器中,它将保持一段时间,但是最后会被JVM的垃圾回收机制干掉。
类加载器是用来把类class装进内存的,类加载器有以下种类
1)引导类加载器(根加载器):c++写的,java平台核心库,用来装载核心类库(RT.JAR),该加载器无法直接获取(爷爷)
2)扩展类加载器:把jre/lin/ext目录下的jar包装入工作库(爸爸)
3)系统加载器:用户自定义的,把一些项目指定的jar包加载进去,最常用(儿子)
测试代码:
1 package reflection; 2 3 public class Test07 { 4 public static void main(String[] args) throws ClassNotFoundException { 5 // 获取 系统类的加载器 6 ClassLoader systemClassLoader = ClassLoader.getSystemClassLoader(); 7 System.out.println("系统类的加载器:" + systemClassLoader); 8 9 // 获取系统类加载器的父类加载器-->扩展类 10 ClassLoader parent = systemClassLoader.getParent(); 11 System.out.println("系统类加载器的父类:" + parent); 12 13 // 获取扩展类加载器的父类加载器-->根加载器,java直接读取不到,返回null 14 ClassLoader parent1 = parent.getParent(); 15 System.out.println("扩展类加载器的父类:" + parent1); 16 17 // 测试当前类是哪个加载器加载的 18 ClassLoader classLoader = Class.forName("reflection.Test07").getClassLoader(); 19 System.out.println("当前类加载 器:" + classLoader); 20 21 // 测试JDK内部的类是哪个加载器加载 22 ClassLoader classLoader1 = Class.forName("java.lang.Object").getClassLoader(); 23 System.out.println("rt包下的jdk内置的类加载器:" + classLoader1); 24 25 // 如何获取系统类加载器可以加载的路径 26 System.out.println(System.getProperty("java.class.path")); 27 28 } 29 }
输出结果:
双亲委派机制:如果你自己定义一个java.lang.String包,类加载器会向上去扩展类找 ,再去根里面找,如果找到了,就不会用你自定义的。