Java通过jar包方式加载指定对象类教程以及源码分析

简介: Java通过jar包方式加载指定对象类教程以及源码分析

一,类加载器基础巩固


类加载器是JVM的重要核心组件之一,也是字节码执行的发源地,只有准确加载了类,JVM才能够创建对象。

一般地有三种类加载器,其名称、对应的对象类以及作用分别是(以JDK8为例):


启动类加载器——加载JRE库文件

用于加载rt.jar等11份文件,如下图所示,

1666254860426.jpg


扩展类加载器——加载JRE扩展文件

用于加载dnsns.jar等12份文件,如下图所示,

1666254871488.jpg


系统类加载器——加载类路径(classPath)下的所有文件


后两种类加载器均是sun.misc.Launcherd的内部嵌套静态类且继承了java.net包中的URLClassLoader类,启动类使用的加载器为null。

1666254885450.jpg


二,使用场景


1.对类进行动态地加载、使用和卸载

常用在Web系统中对编解码器的开发,将编解码器按照一定规则编写好后打包成外部jar包,上传到系统平台中,通过类加载器对jar包的加载完成编解码对象的生成,进而使用其编解码方法。不需要重新编译整个系统程序,完成代码的局部更新;

2.自定义类加载器

通过自定义类加载器,


三,打包一个jar


3.1 功能


把字符串解码为新格式的字符串


3.2 解码接口


public interface DecodeToPOJO {
    Object decode(String hexContent);
}


3.3 解码接口实现类


public class Decoder implements DecodeToPOJO {
    @Override
    public Object decode(String hexContent) {
        return "say Hi";
    }
    public static void main(String[] args) {
        Decoder decoder = new Decoder();
        System.out.println(decoder.decode(""));
    }
}

3.4 目的


在HTTP GET接口中通过加载jar包的形式调用Decoder类的decode方法。


3.5 生成jar包

1666255061227.jpg


四,编写一个接口加载类


    @RequestMapping("/doClassLoad")
    public String doClassLoad() throws IOException {
        long start = System.currentTimeMillis();
        ClassLoader classLoader = JarClassLoader.loadJarToSystemClassLoader(new File("F:\\respository\\MessageTransformer\\out"));
        Class aClass = null;
        try {
            aClass = classLoader.loadClass("org.leobit.codec.Decoder");
            System.out.println(aClass.hashCode());
        } catch (Exception e) {
            e.printStackTrace();
        }
        long end = System.currentTimeMillis();
        return Long.toString(end - start);
    }

JarClassLoader类在cn.hutool.hutool-core包中,省的自己编写了。


<dependency>
  <groupId>cn.hutool</groupId>
  <artifactId>hutool-core</artifactId>
  <version>4.4.0</version>
</dependency>

五,源码分析


我们主要关注加载类方法。

aClass = classLoader.loadClass("org.leobit.codec.Decoder");

进入该方法,可以看到该方法是线程安全的,首选检查类是否被加载,未加载的选择合适的类加载器进行加载,这就是双亲委派机制,

    protected Class<?> loadClass(String name, boolean resolve)
        throws ClassNotFoundException
    {
        synchronized (getClassLoadingLock(name)) {
            // First, check if the class has already been loaded
            Class<?> c = findLoadedClass(name);
            if (c == null) {
                long t0 = System.nanoTime();
                try {
                    if (parent != null) {
                        c = parent.loadClass(name, false);
                    } else {
                        c = findBootstrapClassOrNull(name);
                    }
                } catch (ClassNotFoundException e) {
                    // ClassNotFoundException thrown if class not found
                    // from the non-null parent class loader
                }
                if (c == null) {
                    // If still not found, then invoke findClass in order
                    // to find the class.
                    long t1 = System.nanoTime();
                    c = findClass(name);
                    // this is the defining class loader; record the stats
                    sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);
                    sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
                    sun.misc.PerfCounter.getFindClasses().increment();
                }
            }
            if (resolve) {
                resolveClass(c);
            }
            return c;
        }
    }

重点看findClass方法,实现类是URLClassLoaderfindClass方法后的sun.misc.PerfCounter表示性能计数器,可以看到生成类对象后,对象类计数加1。


protected Class<?> findClass(final String name)
        throws ClassNotFoundException
    {
        final Class<?> result;
        try {
            result = AccessController.doPrivileged(
                new PrivilegedExceptionAction<Class<?>>() {
                    public Class<?> run() throws ClassNotFoundException {
                        String path = name.replace('.', '/').concat(".class");
                        Resource res = ucp.getResource(path, false);
                        if (res != null) {
                            try {
                                return defineClass(name, res);
                            } catch (IOException e) {
                                throw new ClassNotFoundException(name, e);
                            }
                        } else {
                            return null;
                        }
                    }
                }, acc);
        } catch (java.security.PrivilegedActionException pae) {
            throw (ClassNotFoundException) pae.getException();
        }
        if (result == null) {
            throw new ClassNotFoundException(name);
        }
        return result;
    }

在来看defineClass方法,就是最终的生成对象的方法。


  private Class<?> defineClass(String name, Resource res) throws IOException {
        long t0 = System.nanoTime();
        int i = name.lastIndexOf('.');
        URL url = res.getCodeSourceURL();
        if (i != -1) {
            String pkgname = name.substring(0, i);
            // Check if package already loaded.
            Manifest man = res.getManifest();
            definePackageInternal(pkgname, man, url);
        }
        // Now read the class bytes and define the class
        java.nio.ByteBuffer bb = res.getByteBuffer();
        if (bb != null) {
            // Use (direct) ByteBuffer:
            CodeSigner[] signers = res.getCodeSigners();
            CodeSource cs = new CodeSource(url, signers);
            sun.misc.PerfCounter.getReadClassBytesTime().addElapsedTimeFrom(t0);
            return defineClass(name, bb, cs);
        } else {
            byte[] b = res.getBytes();
            // must read certificates AFTER reading bytes.
            CodeSigner[] signers = res.getCodeSigners();
            CodeSource cs = new CodeSource(url, signers);
            sun.misc.PerfCounter.getReadClassBytesTime().addElapsedTimeFrom(t0);
            return defineClass(name, b, 0, b.length, cs);
        }
    }

一直重载定义类的方法,最后终于在defineClass1Native方法中创建出对象。


可以看到就是将文件流读入到虚拟机中进行对象类的创建,该过程又包含class文件的校验、类变量的准备、方法的解析和初始化的阶段。


六,加载jar包结果展示


我们请求第4节中编写的接口,打印每次加载类的哈希码。


可以看到通过同一类加载器加载的对象的哈希码是相同的。

1666255181520.jpg

相关文章
|
2天前
|
Java 关系型数据库 MySQL
Elasticsearch【问题记录 01】启动服务&停止服务的2类方法【及 java.nio.file.AccessDeniedException: xx/pid 问题解决】(含shell脚本文件)
【4月更文挑战第12天】Elasticsearch【问题记录 01】启动服务&停止服务的2类方法【及 java.nio.file.AccessDeniedException: xx/pid 问题解决】(含shell脚本文件)
26 3
|
2天前
|
Java
Java基础之对象的引用
Java基础之对象的引用
5 0
|
6天前
|
Java
Java中如何克隆一个对象?
【4月更文挑战第13天】
14 0
|
7天前
|
Java 编译器
Java Character 类
4月更文挑战第13天
|
7天前
|
Java API 数据库
深入解析:使用JPA进行Java对象关系映射的实践与应用
【4月更文挑战第17天】Java Persistence API (JPA) 是Java EE中的ORM规范,简化数据库操作,让开发者以面向对象方式处理数据,提高效率和代码可读性。它定义了Java对象与数据库表的映射,通过@Entity等注解标记实体类,如User类映射到users表。JPA提供持久化上下文和EntityManager,管理对象生命周期,支持Criteria API和JPQL进行数据库查询。同时,JPA包含事务管理功能,保证数据一致性。使用JPA能降低开发复杂性,但需根据项目需求灵活应用,结合框架如Spring Data JPA,进一步提升开发便捷性。
|
8天前
|
存储 Java
Java基础教程(7)-Java中的面向对象和类
【4月更文挑战第7天】Java是面向对象编程(OOP)语言,强调将事务抽象成对象。面向对象与面向过程的区别在于,前者通过对象间的交互解决问题,后者按步骤顺序执行。类是对象的模板,对象是类的实例。创建类使用`class`关键字,对象通过`new`运算符动态分配内存。方法包括构造函数和一般方法,构造函数用于对象初始化,一般方法处理逻辑。方法可以有0个或多个参数,可变参数用`类型...`定义。`this`关键字用于访问当前对象的属性。
|
11天前
|
Java Shell
Java 21颠覆传统:未命名类与实例Main方法的编码变革
Java 21颠覆传统:未命名类与实例Main方法的编码变革
13 0
|
11天前
|
Java
Java 15 神秘登场:隐藏类解析未知领域
Java 15 神秘登场:隐藏类解析未知领域
15 0
|
11天前
|
存储 Java 编译器
对象的交响曲:深入理解Java面向对象的绝妙之处
对象的交响曲:深入理解Java面向对象的绝妙之处
46 0
对象的交响曲:深入理解Java面向对象的绝妙之处
|
13天前
|
安全 Java
append在Java中是哪个类下的方法
append在Java中是哪个类下的方法
21 9