Fork me on GitHub

【JAVA反射】自定义类加载器

为什么要自定义类加载器

个人认为是由于我们的应用程序需要加载一些类资源,这些资源有可能来源于网络或非当前应用程序目录下的其他磁盘,所以就要类加载器来动态加载这些类文件。
我觉得这是不是可以实现类似的热部署,在不重启服务情况下,实现类文件的自由替换,

原理

通过流程读取二进制类文件,再通过ClassLoader中的方法Class<?> defineClass(String name, byte[] b, int off, int len)来获取Class,再通过反射获取类实例以及通过Method的invoke就去回调用类中定义的方法。此时不的不说反射的强大,哈哈!

这个类加载器有个启动顺序一定要注意
启动类加载器--->扩展类加载器--->应用程序类加载器--->自定义类加载器

  • 启动类加载器:加载JAVA_HOME/lib下的类
  • 扩展类加载器:加载JAVA_HOME/lib/ext下的类
  • 应用程序类加载器:加载java -D java.class.path 指定路径下的类库,也就是classpath路径
  • 自定义类加载器: 请看下文

实例

  • 新建自定义加载器类MyClassLoader,并继承类ClassLoader
package com.icodesoft;

import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.InputStream;

public class MyClassLoader extends ClassLoader {

    // 获取类
    public Class<?> loadClass(String className, String classPath) throws Exception {
        byte[] bytes = this.loadData(classPath);
        if (bytes.length > 0) {
            return super.defineClass(className, bytes, 0, bytes.length);
        }
        return null;
    }

    // 获取二进制类文件内容
    private byte[] loadData(String classPath) throws Exception {
        InputStream inputStream = null;
        ByteArrayOutputStream byteArrayOutputStream = null;

        try {
            byteArrayOutputStream = new ByteArrayOutputStream();
            File file = new File(classPath);
            byte[] data = new byte[(int) file.length()];
            inputStream = new FileInputStream(file);
            inputStream.read(data);
            byteArrayOutputStream.write(data);
            return byteArrayOutputStream.toByteArray();
        }finally {
            if (inputStream != null) inputStream.close();
            if (byteArrayOutputStream != null) byteArrayOutputStream.close();
        }
    }
}
  • 新建程序入口类 DemoApplication(用于测试)
package com.icodesoft;
import java.lang.reflect.Method;

public class DemoApplication {

    public static void main(String[] args) throws Exception {
        Class<?> myClass = new MyClassLoader().loadClass("com.icodesoft.test.Hello", "d:/SourceCode/Hello.class");
        Object o = myClass.getDeclaredConstructor().newInstance();
        Method say = myClass.getMethod("say");
        say.invoke(o);
        System.out.println("====>" + myClass);
    }
}
  • 在D盘的SourceCode文件夹下创建实例中被加载的类Hello.java, 并定义方法say
package com.icodesoft.test;

public class Hello {
    public void say() {
        System.out.println("Hello World");
    }
}
  • 编译通过执行命令javac Hello.java生成Hello.class
D:\SourceCode>javac Hello.java

测试结果如下,已成功调用Hello类中的say方法

posted @ 2020-09-29 17:06  逍遥メ风  阅读(242)  评论(0编辑  收藏  举报