一、概述
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()方法传入参数