为什么要自定义类加载器
个人认为是由于我们的应用程序需要加载一些类资源,这些资源有可能来源于网络或非当前应用程序目录下的其他磁盘,所以就要类加载器来动态加载这些类文件。
我觉得这是不是可以实现类似的热部署,在不重启服务情况下,实现类文件的自由替换,
原理
通过流程读取二进制类文件,再通过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