一、概述
1、自定义的类加载器的必须继承ClassLoader
2、loadClass方法与findClass方法
一般覆盖findClass()方法,该方法自己将类加载至jvm。loadClass()方法不用覆盖,是一套调用父委托机制完整流程。
3、defineClass方法
传入一个byte[]数组,返回一个Class对象。
二、关键点
1、编写一个程序调用类加载器加载类,在源程序中不能用该类名定义引用变量,因为编译器无法识别这个类。程序中可以除了使用ClassLoader.load方法之外,还可以使用设置线程的上下文类加载器或者系统类加载器,然后再使用Class.forName
2、用加密后的类文件替换CLASSPATH环境下的类文件,再执行上一步操作就出问题了,错误说明是AppClassLoader类装载器装载失败
3、删除CLASSPATH环境下的类文件,再执行上一步操作就没问题了
三 、代码说明
1、MyClassLoader.java
package staticimport.classloader.custom;
import java.io.ByteArrayOutputStream;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.InputStream;
import java.io.OutputStream;
/***
*
* 编写和测试自己编写的解密类加载器
*
*
* @author Liu
*
*/
public class MyClassLoader extends ClassLoader {
private String classPath;
public MyClassLoader(String classPath) {
super();
this.classPath = classPath;
}
public MyClassLoader() {
super();
}
public MyClassLoader(ClassLoader parent) {
super(parent);
}
@SuppressWarnings("deprecation")
@Override
protected Class<?> findClass(String name) throws ClassNotFoundException {
try {
// String srcPath = classPath + "\\" + name + ".class";
String srcPath = classPath + "\\" + name.substring(name.lastIndexOf(".") + 1) + ".class";
InputStream is = new FileInputStream(srcPath);
ByteArrayOutputStream os = new ByteArrayOutputStream();
cypher(is, os);
is.close();
byte[] bytes = os.toByteArray();
os.close();
// return defineClass(name, bytes, 0, bytes.length);
return defineClass(bytes, 0, bytes.length);
} catch (Exception e) {
e.printStackTrace();
}
return super.findClass(name);
}
public static void main(String[] args) throws Exception {
String srcPath = args[0];
String destDir = args[1];
String fileName = srcPath.substring(srcPath.lastIndexOf("\\") + 1);
String destPath = destDir + "\\" + fileName;
InputStream is = new FileInputStream(srcPath);
OutputStream os = new FileOutputStream(destPath);
cypher(is, os);
is.close();
os.close();
}
// 字节加解密算法
private static void cypher(InputStream is, OutputStream os) throws Exception {
int b = -1;
while ((b = is.read()) != -1) {
os.write(b ^ 0xFFFF);
}
}
}
2、主类调用代码
Class clazz = new MyClassLoader("cllib").loadClass("staticimport.classloader.custom.ClassLoaderAttachment");
//ClassLoaderAttachment继承Date类型的原因,跳过编译器检查!
Date date = (Date) clazz.newInstance();
System.out.println(date);
注:
测试加密算法需配置MyClassLoader.java的main()方法传入参数