JVM 类加载器

类加载器
VM 中有多个类加载器,分饰不同的角色。每个类加载器由它的父加载器加载。bootstrap 加载器除外,它是所有最顶层的类加载器。
 
Bootstrap 加载器
使用C/C++语言实现的,因为它在 JVM 加载以后的早期阶段就被初始化了。bootstrap 加载器负责载入基础的 Java API,比如包含 rt.jar。它只加载拥有较高信任级别的启动路径下找到的类,因此跳过了很多普通类需要做的校验工作。
  • C/C++语言实现的
  • 它用来加载核心库(JAVA_HOME/jre/lib/rt.jar,resources.jar,或sun.boot.class.path路径下的内容),用于提供JVM自身需要的类
  • 并不继承自java.lang.ClassLoader,没有父加载器;
  • 加载扩展类加载器和系统类加载器,并指定为它们的父类加载器;
  • 出于安全考虑,Bootstrap只加载包名为java、javax、sum开头的类;

 

Extension 加载器加载了标准 Java 扩展 API 中的类,比如 security 的扩展函数。
  • Java语言编写,由sum.misc.Launcher$ExtClassLoader实现
  • 派生于ClassLoader类
  • 父类加载器为启动类加载器
  • 从java.ext.dirs系统属性所指定的目录中加载类库,或从JDK的安装目录的jre/lib/ext子目录(扩展目录)下加载类库。如果用户创建的jar放在此目录下,也会自动由扩展类加载器加载
 
System 加载器是应用的默认类加载器,比如从 classpath 中加载应用类,也称之为应用程序类加载器
  • Java语言编写,由sum.misc.Launcher$AppClassLoader实现
  • 派生于ClassLoader类
  • 父类加载器为启动类加载器
  • 它负责加载环境变量classpath或系统属性 java.class.path 指定路径下睥类库
  • 该类加载是程序中默认的类加载器,一般来说,Java应用的类都是由它来完成加载 
  • 通过ClassLoader#getSystemClassLoader()方法可以获取到该类加载器
 
下面示例说明Java应用的类都是由它来完成系统类加载器加载的 
public class Test {
    public static void main(String[] args) throws ClassNotFoundException {
        new Test().test();
    }    
    public void test() throws ClassNotFoundException {
        // Java应用的类都是由它来完成加载 
        System.out.println(this.getClass().getClassLoader());    // sun.misc.Launcher$AppClassLoader@2a139a55
        System.out.println(this.getClass().getClassLoader().getParent());// sun.misc.Launcher$ExtClassLoader@7852e922
        // 编译生成的.class文件的bin目录
        String path = System.getProperty("java.class.path"); 
    }
}

 

用户自定义类加载器也可以用来加载应用类。使用自定义的类加载器有很多特殊的原因:运行时重新加载类或者把加载的类分隔为不同的组,典型的用法比如 web 服务器 Tomcat。
为什么要自定义类加载器?
  • 隔离加载类
  • 修改类加载的方式
  • 扩展加载源
  • 防止源码泄露

 

关于系统类加载器与自定义加载器注意以下几点:

MyClassLoader 为自定义类加载器,相关代码,自已可以百度一下。

(1)一个类加载器实例可以加载多个类

// myClassLoader1 为自定义类加载器实例
MyClassLoader myClassLoader1 = new MyClassLoader();
Class<?> clazz1 = myClassLoader1.findClass("com.classloader.PrintServiceImpl");
Class<?> clazz3 = myClassLoader1.findClass("com.classloader.ClassFile");

 

(2)但是对于同一个类型,一个类加载器实例只能加载一次,下面这个就会抛出异常:

Exception in thread "main" java.lang.LinkageError: loader (instance of  com/classloader/MyClassLoader): attempted  duplicate class definition for name: "com/classloader/PrintServiceImpl" at java.lang.ClassLoader.defineClass1(Native Method)

MyClassLoader myClassLoader1 = new MyClassLoader();
Class<?> clazz1 = myClassLoader1.findClass("com.classloader.PrintServiceImpl");
Class<?> clazz2 = myClassLoader1.findClass("com.classloader.PrintServiceImpl");

异常:一个本地方法异常;LinkageError:

 

(3)对于同一个类型若要加载多次,只能通过不同的类加载器的实例来加载 

添加虚拟机参数:-verbose:class

MyClassLoader myClassLoader1 = new MyClassLoader();
Class<?> clazz1 = myClassLoader1.findClass("com.classloader.PrintServiceImpl");
MyClassLoader myClassLoader2 = new MyClassLoader();
Class<?> clazz2 = myClassLoader2.findClass("com.classloader.PrintServiceImpl");
System.out.println(clazz1==clazz2);    // false
// 因为myClassLoader1 = clazz1.getClassLoader(),所以下面打印必为false ;
System.out.println(clazz1.getClassLoader()==clazz2.getClassLoader());// false
// myClassLoader1.getClass()的值为java.lang.Class,所以下面必为true
System.out.println(clazz1.getClassLoader().getClass()==clazz2.getClassLoader().getClass());// true
// myClassLoader1.getParent();
System.out.println(clazz1.getClassLoader().getParent()==clazz2.getClassLoader().getParent());// true

 

(4)Class.forName("")加载一次(并且会执行初始化操作)

Class.forName("com.classload.Student");
Class<?> clazz1 = Class.forName("com.classload.Student");
Class<?> clazz2 = Class.forName("com.classload.Student");
System.out.println(Class.forName("com.classload.Student")==clazz1);// true
System.out.println(clazz1);            // class com.classload.Student
System.out.println(clazz2);            // class com.classload.Student
System.out.println(clazz1==clazz2);    // true

 

posted @ 2020-04-20 02:51  cao_xiaobo  阅读(224)  评论(1编辑  收藏  举报