JVM执行时会通过设置的CLASSPATH环境变量进行指定路径的字节码文件查找,加载字节码文件的操作就需要用到类加载器。
代码示例
package com.xsry.classloader;
import java.io.*;
//自定义需要继承CLassLoader类
public class ClassLoadDiv extends ClassLoader {
//此处使用时换成你的字节码文件路径
private static final String PATH = "E:" + File.separator + "javapage" + File.separator + "DivClassLoader.class";
/**
* @param name 类的完整名称 “包.类”
* @return 返回一个指定类的class对象
* @throws Exception 如果类文件不存在则无法加载
*/
public Class<?> loadData(String name) throws Exception {
byte data[] = this.loadLocalClass();
if(data != null)
//将加载的二进制数据文件转换为class的处理
return super.defineClass(name, data, 0, data.length);
return null;
}
private byte[] loadLocalClass() throws Exception {
InputStream ins = null;
ByteArrayOutputStream bos = null;
byte data[] = null;
try {
bos = new ByteArrayOutputStream(); //将数据载入到内存中
ins = new FileInputStream(new File(PATH));
ins.transferTo(bos);
data = bos.toByteArray();
} catch (Exception e) {
e.printStackTrace();
} finally {
if(ins != null)
ins.close();
if(bos != null)
bos.close();
}
return data;
}
}
外部字节码文件源代码
package xsry.div.ClassLoader;
public class DivClassLoader {
public void send() {
System.out.println("Hello ClassLoader!");
}
}
主程序
import java.lang.reflect.Method;
//自定义类加载器
//使用类加载器可以将部分经常更新的class类文件放在服务器中,方便程序的维护,使开发具有极大的灵活性
public class Demo1 {
public static void main(String[] args) throws Exception {
ClassLoadDiv cld = new ClassLoadDiv();
Class<?> cls = cld.loadData("xsry.div.ClassLoader.DivClassLoader");
//类不在CLASSPATH中,无法直接将对象转为指定类型,只能通过反射调用
Object obj = cls.getDeclaredConstructor().newInstance();
Method mth = cls.getDeclaredMethod("send");
mth.invoke(obj);
}
}
“双亲加载”:如果定义类与系统类重复,那么自定义类将不会被加载