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

3ce3119ff279599f973fe610120be282.png

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

63e4ceeef84f91bccc090033cb88c430.png

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值