zoukankan      html  css  js  c++  java
  • 【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方法

  • 相关阅读:
    2019春总结
    2019春第一周编程总结
    第十二周
    2019春第七周编程总结
    2019春第四周作业
    人生路上对我影响最大的三位老师
    第五周课程总结&试验报告(三)
    第四周课程总结&试验报告(二)
    第三周课程总结&实验报告一
    第二周java学习总结
  • 原文地址:https://www.cnblogs.com/bjxly/p/13750621.html
Copyright © 2011-2022 走看看