JVM类加载器

在类加载过程的加载阶段,有一个通过类的全限定名获取描述类的二进制流的动作,Java虚拟机设计团队有意将这个动作的实现放到虚拟机之外实现,以便让用户自己决定如何获取所需类,这个动作的实现被称为类加载器。JVM根据职能的不同,设计了以下四种类加载器:

  • 引导类加载器(BootStrap ClassLoader)
  • 扩展类加载器(Extension ClassLoader)
  • 应用程序类加载器(Application ClassLoader)
  • 自定义类加载器(User ClassLoader)

前三种是虚拟机自带的类加载器,自定义类加载器是用户根据自己的需求设计的类加载器,下图是四种类加载器之间的关系,图中所表示的层次关系,并非类的继承关系,而是描述类加载器协作关系,通常这个协作关系通过组合的方式实现(设计模式中的组合,通过组合复用父类加载器的代码,除了引导类加载器,其他类加载器都有父类加载器),这个协作动作的实现被称为"双亲委派模型",后面的篇章中单独讲。
在这里插入图片描述
在Java程序中可按照如下代码获取除引导类加载器外的各个类加载器。

public class ClassLoaderTest {
    public static void main(String[] args) {
        // (启动类)系统类加载器:
        ClassLoader systemClassLoader = ClassLoader.getSystemClassLoader();
        System.out.println(systemClassLoader); //sun.misc.Launcher$AppClassLoader@18b4aac2
        //扩展类加载器:
        ClassLoader extendClassLoader = systemClassLoader.getParent();
        System.out.println(extendClassLoader); //sun.misc.Launcher$ExtClassLoader@1b6d3586
        // 引导类加载器:
        ClassLoader bootstrapClassLoader = extendClassLoader.getParent();
        System.out.println(bootstrapClassLoader); // null
         // 用户自定义的类默认用系统类加载器
        ClassLoader classLoader = ClassLoaderTest.class.getClassLoader();
        System.out.println(classLoader); // sun.misc.Launcher$AppClassLoader@18b4aac2
    }
}

引导类加载器

引导类加载器是由C/C++语言实现,嵌套在虚拟机内部,用来加载java核心库中的类,特性如下:

  1. 只加载JAVA_HOME/jre/lib/rt.jar、resources.jar或sun.boot.class.path路径下的内容
  2. 加载扩展类和应用程序类类加载器,并且是它们的父类加载器,不继承ClassLoader类,其本身没有父类加载器。
  3. 出于安全考虑,只加载包名为java、javax、sun开头的类
  4. 引导类加载器无法通过getClassLoader()方法获取,只会返回null,如String.class.getClassLoader()。

扩展类加载器

扩展类加载器是Java语言实现的类加载器,在sun.misc.Launcher.ExtClassLoader中实现,派生于ClassLoader类,特性:

  1. 本身先有引导类加载器加载,父类加载器为引导类加载器(非继承关系)。
  2. 加载java.ext.dirs系统属性所指定的目录中的类库,或者加载JDK安装目录的jre/ext/dir扩展目录下的类库,用户可以将自己的jar放入该目录,通过扩展类加载器加载。

应用程序类加载器

应用程序类加载器也是Java语言实现的类加载器,在sun.misc.Launcher.AppClassLoader中实现,派生于ClassLoader类,特性:

  1. 本身先有引导类加载器加载,父类加载器为扩展类加载器(非继承关系)。
  2. 加载环境变量classpath或者java.class.path属性指定的目录下的类库。
  3. 程序中默认的类加载器,一般的Java应用程序都由它加载,可以通过java.lang.ClassLoader#getSystemClassLoader方法获取。

用户自定义类加载器

对于某一些特殊的类加载需求,用户可以通过继承ClassLoader实现自定义的类加载器,通过自定义类加载器,可以在以下的需求场景使用:

  1. 隔离类,如类路径冲突。
  2. 防反编译加密Class文件。
  3. 扩展类的加载源。

自定义实现类加载器可以通过继承java.lang.ClassLoader并重写loadClas()方法或者实现findClass()(推荐)方法,如下代码就是加载指定目录class的简单实现。

public class DirClassLoader extends ClassLoader {

    private String dir;  // 指定目录
    public DirClassLoader(String dir) {
        this.dir = dir;
    }
    @Override
    protected Class<?> findClass(String name) throws ClassNotFoundException {
        // class文件路径
        String classPath = dir + File.separator + name + ".class";
        try (InputStream is = new FileInputStream(classPath)){
            byte[] b = new byte[is.available()];
            is.read(b);
            return defineClass(name, b, 0, b.length);
        } catch (Exception ex) {
            throw new ClassNotFoundException();
        }
    }

    public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException {
        DirClassLoader dirClassLoader = new DirClassLoader("D:\\setup");
        Object obj = dirClassLoader.loadClass("Hello").newInstance();
        System.out.println(obj.getClass());
        System.out.println(obj.getClass().getClassLoader());
    }
}

类在虚拟机中的唯一性

在虚拟机中,每个类的唯一性,由类和类加载器两个因素决定,也就是即使同一个类被不同类加载器加载,也会导致这两个类不相同。在上篇文章说过,类加载完成以后会在堆区生成一个类的java.lang.Class对象作为外部接口访问方法区对应的类信息,这个操作具象化就是Object的getClass()方法。getClass()获取到类的Class对象,从而根据getClassLoader()方法获取类加载器,下面代码中自定义了类加载器MyClassLoader,在main中通过自定义类加载器加载一次ClassLoaderUniqueTest类并创建实例,同时在执行main方法时,虚拟机会通过应用程序类加载器加载一次ClassLoaderUniqueTest类,最后通过instanceof比较类型,这个比较类型并不限于instanceof,还可以用equals。

public class ClassLoaderUniqueTest {
    // 自定义类加载器
    static class MyClassLoader extends ClassLoader {
        @Override
        public Class<?> loadClass(String name) throws ClassNotFoundException {
            try {
                String fileName = name.substring(name.lastIndexOf(".") + 1) + ".class";
                InputStream is = this.getClass().getResourceAsStream(fileName);
                if (is == null) {
                    return super.loadClass(name);
                }
                byte[] b = new byte[is.available()];
                is.read(b);
                return defineClass(name, b, 0, b.length);
            } catch (IOException e) {
                throw new ClassNotFoundException();
            }
        }
    }

    public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException {
        ClassLoader myClassLoader = new MyClassLoader();
        Object obj = myClassLoader.loadClass("classloder.ClassLoaderUniqueTest").newInstance();
        System.out.println(obj.getClass().getClassLoader());
        System.out.println(obj instanceof classloder.ClassLoaderUniqueTest);
    }
}

执行结果

classloder.ClassLoaderUniqueTest$MyClassLoader@74a14482
false
  • 20
    点赞
  • 19
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
JVM的类加载是由类加载器及其子类实现的。类加载器是Java运行时系统的重要组成部分,负责在运行时查找和加载类文件中的类。在JVM中,类加载器按照一定的层次结构进行组织,每个类加载器负责加载特定位置的类。其中,启动类加载器(Bootstrap ClassLoader)是负责加载存放在<JAVA_HOME>/lib目录中的核心类库,如rt.jar、resources.jar等,同时也可以加载通过-Xbootclasspath参数指定的路径中的类库。启动类加载器是用C语言编写的,随着JVM启动而加载。当JVM需要使用某个类时,它会通过类加载器查找并加载这个类。加载过程会经历连接阶段,包括验证、准备和解析。在验证阶段,JVM会确保加载的类信息符合JVM规范。在准备阶段,JVM会为类变量分配内存并设置初始值,在方法区中分配这些内存。在解析阶段,JVM会根据符号引用替换为直接引用,以便后续的使用。通过类加载器的协同工作,JVM能够在运行时动态加载类,从而实现Java的灵活性和跨平台性。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *3* [JVM 的类加载原理](https://blog.csdn.net/ChineseSoftware/article/details/119212339)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] - *2* [JVM类加载器](https://blog.csdn.net/rockvine/article/details/124825354)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值