ClassLoader类加载器 | 带你学《Java语言高级特性》之九十四

简介: 在Java语言中提供了一个系统的环境变量:CLASSPATH,这个环境属性的作用主要是在JVM进程启动时进行类加载路径的定义,在JVM中可以根据类加载器而后进行指定路径中类的加载,也就是说找到了类的加载器就意味着找到了类的来源。

上一篇:级联对象实例化 | 带你学《Java语言高级特性》之九十三
【本节目标】
本节介绍了ClassLoader类加载器,其中自定义的类加载器其加载的顺序是在所有系统类加载器的最后。系统类中的类加载器都是根据CLASSPATH路径进行加载的,而如果有了自定义的类加载器,就可以由开发者任意指定类的加载位置。

ClassLoader类加载器

在Java语言中提供了一个系统的环境变量:CLASSPATH,这个环境属性的作用主要是在JVM进程启动时进行类加载路径的定义,在JVM中可以根据类加载器而后进行指定路径中类的加载,也就是说找到了类的加载器就意味着找到了类的来源。

image.png
ClassLoader

系统类加载器

如果现在要想获得类的加载器,那么一定要通过ClassLoader来获取,而要想获取ClassLoader类的对象,则必须利用Class类(反射的根源)实现,方法:
public ClassLoader getClassLoader()
当获取了ClassLoader后还可以获取其父类的ClassLoader类对象:
public final ClassLoader getParent​()
范例:观察类加载器

class Message{}
public class JavaAPIDemo {
    public static void main(String[] args)throws Exception{
        Class<?> clazz=Message.class;
        System.out.println(clazz.getClassLoader());//获取当前类加载器
        //1.8:sun.misc.Launcher$AppClassLoader@6659c656
        //1.9+:jdk.internal.loader.ClassLoaders$AppClassLoader@4f8e5cde
        System.out.println(clazz.getClassLoader().getParent());//获取父类加载器
        //1.8:sun.misc.Launcher$ExtClassLoader@60e53b93
        //1.9+:jdk.internal.loader.ClassLoaders$PlatformClassLoader@5d3411d
        System.out.println(clazz.getClassLoader().getParent().getParent());//获取祖父类加载器 ,null
    }
}

从JDK1.8之后的版本(JDK1.9,JDK1.10)提供有一个“PlatformClassLoader”类加载器,而在JDK1.8及以前的版本中提供的加载器为“ ExtClassLoader”,因为在JDK的安装目录中提供了一个ext的目录,开发者可以将*.jar文件拷贝到此目录中,这样就可以直接执行了,但是这样的处理开发并不安全,最初的时候也是不提倡使用的,所以从JDK1.9开始将其彻底废除了,同时为了与系统类加载器和应用类加载器之间保持设计的平衡,提供有平台类加载器。
image.png

当你获得了类加载器后就可以利用类加载器来实现类的反射加载处理:
protected Class<?> findClass​(String name) throws ClassNotFoundException

自定义类加载器

清楚了类加载器的功能后就可以根据自身的需求来实现自定义的类加载器,但是千万要记住一点:自定义的类加载器其加载的顺序是在所有系统类加载器的最后。系统类中的类加载器都是根据CLASSPATH路径进行加载的,而如果有了自定义的类加载器,就可以由开发者任意指定类的加载位置。

image.png
自定义类加载器

1、随意编写一个程序类,并且将这个类保存在磁盘上。

public class Message {
    public void send(){
        System.out.println("www.mldn.cn");
    }
}

2、将此类直接拷贝到系统磁盘上(非项目路径)进行编译处理,并且不打包:javac Message.java,此时并没有进行打包处理,所以这个类无法通过CLASSPATH正常加载。
javac /Users/david/Documents/mydir/Message.java
3、自定义一个类加载器,并且继承自ClassLoader类。在ClassLoader类中提供有一个字节转换为类结构的方法:

定义类:
protected final Class<?> defineClass​(String name, byte[] b, int off, int len) throws ClassFormatError

import java.io.*;
public class MLDNClassLoader extends ClassLoader {
    private static final String MESSAGE_CLASS_PATH = "D:" + File.separator + "Message.class";
    /**
     * 进行指定类的加载
     *
     * @param className 类的完整名称“包.类”
     * @return 返回一个指定类的Class对象
     * @throws Exception 如果类文件不存在,则无法加载
     */
    public Class<?> loadData(String className) throws Exception {
        byte[] data = loadClassData();//读取二进制数据文件
        if (data != null) {  //读取到了
            return super.defineClass(className, data, 0, data.length); 
        }
        return null;
    }
    private byte[] loadClassData() throws Exception {//通过文件进行类的加载
        InputStream input = null;
        ByteArrayOutputStream bos = null;  //将数据加载到内存之中
        byte data [] = null;
        try { 
            bos = new ByteArrayOutputStream();  //实例化内存流
            input = new FileInputStream(new File(MESSAGE_CLASS_PATH));  //文件加载流
            input.transferTo(bos);  //读取数据
            data = bos.toByteArray();  //将所有读取到的字节数取出
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            if (input != null) {
                input.close();
            }
            if (bos != null) {
                bos.close();
            }
        }
        return data;
    }
}

4、编写测试类实现类加载控制。

import  java.lang.reflect.Method;
import cn.mldn.util.MLDNClassLoader;
public class JavaAPIDemo {
    public static void main(String[] args)throws Exception{
        MLDNClassLoader classLoader = new MLDNClassLoader();  //实例化自定义类加载器
        Class<?> cls=classLoader.loadData("cn.mldn.util.Message");
        Object obj = cls.getDeclaredConstructor().newInstance();
        Method method = cls.getDeclaredMethod("send");
        method.invoke(obj);
    }
}

如果在以后结合网络程序开发的话,就可以通过一个远程的服务器来确定一个类的功能。

image.png
应用项目

5、观察当前的Message类的加载器的情况

public class JavaAPIDemo {
    public static void main(String[] args)throws Exception{
        MLDNClassLoader classLoader = new MLDNClassLoader();//实例化自定义类加载器
        Class<?> clazz=classLoader.loadData("cn.mldn.util.Message");
        System.out.println(cls.getClassLoader());  //cn.mldn.util.MLDNClassLoader@6979e8cb
        System.out.println(cls.getClassLoader().getParent()); //jdk.internal.loader.ClassLoaders$AppClassLoader@6659c656
        System.out.println(cls.getClassLoader().getParent().getParent());  //jdk.internal.loader.ClassLoaders$PlatformClassLoader@763d9750
    }
}

如果现在定义了一个类:java.lang.String,并且利用了自定义的类加载器进行加载处理,这个类将不会被加载,Java之中针对于类加载器提供有双亲加载机制,如果现在要加载的程序类是由系统提供的类则会由系统类加载器进行加载,在开发者定义的类与系统类名称相同,那么为了保证系统的安全性绝对不会加载。

想学习更多的Java的课程吗?从小白到大神,从入门到精通,更多精彩不容错过!免费为您提供更多的学习资源。
本内容视频来源于阿里云大学

下一篇:静态代理设计模式 | 带你学《Java语言高级特性》之九十五
更多Java面向对象编程文章查看此处

相关文章
|
3月前
|
存储 人工智能 算法
数据结构与算法细节篇之最短路径问题:Dijkstra和Floyd算法详细描述,java语言实现。
这篇文章详细介绍了Dijkstra和Floyd算法,这两种算法分别用于解决单源和多源最短路径问题,并且提供了Java语言的实现代码。
102 3
数据结构与算法细节篇之最短路径问题:Dijkstra和Floyd算法详细描述,java语言实现。
|
2月前
|
监控 Java API
如何使用Java语言快速开发一套智慧工地系统
使用Java开发智慧工地系统,采用Spring Cloud微服务架构和前后端分离设计,结合MySQL、MongoDB数据库及RESTful API,集成人脸识别、视频监控、设备与环境监测等功能模块,运用Spark/Flink处理大数据,ECharts/AntV G2实现数据可视化,确保系统安全与性能,采用敏捷开发模式,提供详尽文档与用户培训,支持云部署与容器化管理,快速构建高效、灵活的智慧工地解决方案。
|
3月前
|
存储 安全 Java
Java Map新玩法:探索HashMap和TreeMap的高级特性,让你的代码更强大!
【10月更文挑战第17天】Java Map新玩法:探索HashMap和TreeMap的高级特性,让你的代码更强大!
86 2
|
2月前
|
SQL 安全 Java
安全问题已经成为软件开发中不可忽视的重要议题。对于使用Java语言开发的应用程序来说,安全性更是至关重要
在当今网络环境下,Java应用的安全性至关重要。本文深入探讨了Java安全编程的最佳实践,包括代码审查、输入验证、输出编码、访问控制和加密技术等,帮助开发者构建安全可靠的应用。通过掌握相关技术和工具,开发者可以有效防范安全威胁,确保应用的安全性。
58 4
|
3月前
|
存储 Java API
优雅地使用Java Map,通过掌握其高级特性和技巧,让代码更简洁。
【10月更文挑战第19天】本文介绍了如何优雅地使用Java Map,通过掌握其高级特性和技巧,让代码更简洁。内容包括Map的初始化、使用Stream API处理Map、利用merge方法、使用ComputeIfAbsent和ComputeIfPresent,以及Map的默认方法。这些技巧不仅提高了代码的可读性和维护性,还提升了开发效率。
118 3
|
3月前
|
存储 安全 Java
Java Map新玩法:深入探讨HashMap和TreeMap的高级特性
【10月更文挑战第19天】Java Map新玩法:深入探讨HashMap和TreeMap的高级特性,包括初始容量与加载因子的优化、高效的遍历方法、线程安全性处理以及TreeMap的自然排序、自定义排序、范围查询等功能,助你提升代码性能与灵活性。
32 2
|
3月前
|
Java 程序员 编译器
在Java编程中,保留字(如class、int、for等)是具有特定语法意义的预定义词汇,被语言本身占用,不能用作变量名、方法名或类名。
在Java编程中,保留字(如class、int、for等)是具有特定语法意义的预定义词汇,被语言本身占用,不能用作变量名、方法名或类名。本文通过示例详细解析了保留字的定义、作用及与自定义标识符的区别,帮助开发者避免因误用保留字而导致的编译错误,确保代码的正确性和可读性。
67 3
|
3月前
|
移动开发 Java 大数据
深入探索Java语言的核心优势与现代应用实践
【10月更文挑战第10天】深入探索Java语言的核心优势与现代应用实践
116 4
|
3月前
|
消息中间件 分布式计算 Java
大数据-73 Kafka 高级特性 稳定性-事务 相关配置 事务操作Java 幂等性 仅一次发送
大数据-73 Kafka 高级特性 稳定性-事务 相关配置 事务操作Java 幂等性 仅一次发送
43 2
|
3月前
|
消息中间件 存储 Java
大数据-58 Kafka 高级特性 消息发送02-自定义序列化器、自定义分区器 Java代码实现
大数据-58 Kafka 高级特性 消息发送02-自定义序列化器、自定义分区器 Java代码实现
68 3