自定义类加载器加载指定目录下的类

package com.itheima.classloader;

import java.io.BufferedInputStream;
import java.io.ByteArrayOutputStream;
import java.io.FileInputStream;
import java.io.IOException;
import java.lang.reflect.Constructor;

public class MyClassLoader extends ClassLoader {
	public static void main(String[] args) throws Exception {
		//通过自定义类加载器加载类,(任意指定目录和名称)
		Class<?> clazz = new MyClassLoader("ClassLoaderLib").loadClass("Person");
		//获取构造方法
		Constructor<?> con = clazz.getConstructor(String.class, int.class);
		//创建对象,成功加载则执行静态代码块中的打印语句打印信息
		Object obj = con.newInstance("张三",15);
		//打印该类的类加载器
		System.out.println(obj.getClass().getClassLoader());
	}

	//classDir记录类所在目录
	private String classDir;

	/**
	 * 复写findClass方法
	 * 因为自定义的类加载器MyClassLoader继承了ClassLoader类,继承了其所有方法
	 * 通过loadClass(String name)方法加载类的时候:
	 * 1、会检查该类是否已经加载,如果没有,会依次委托MyClassLoader的父类、父类的父类
	 *    等类加载器依次尝试加载,都失败则执行findClass(String name)试图获取待加载的类的Class对象
	 * 2、而ClassLoader中findClass(String name)方法直接抛出ClassNotFoundException(name)
	 * 3、因此复写findClass(String name)方法就是以自定义的方式获取类(在父类都无法加载的情况)
	 * 4、通过将某个.class文件读入一个byte类型的数组中,然后通过defineClass方法
	 *    获取该字节码文件(Class对象)并返回
	 * 5、因此,当1中的情况出现时,本类即可通过findClass(String name)获取Class对象给
	 *    loadClass(String name)方法,使其返回同一个Class对象
	 * 	    
	 *    本类主函数内:
	 *    new MyClassLoader("ClassLoaderLib").loadClass("Person");
	 *    new MyClassLoader("ClassLoaderLib").findClass("Person");
	 * 
	 *    因为protected权限修饰符,其他类调用本类加载某个类只能用loadClass方法
	 * 
	 */
	@Override
	protected Class<?> findClass(String name) throws ClassNotFoundException {
		//获取字节码文件的完整路径
		String classPath = classDir + "\\" + name + ".class";
		//定义Class对象引用
		Class<?> clazz = null;
		//设置字节输入流引用
		BufferedInputStream bos = null;
		//定义字节数组输出流引用
		ByteArrayOutputStream baos = null;
		try {
			//建立字节输入流
			bos = new BufferedInputStream(new FileInputStream(classPath)); 
			//定义缓冲数组buf,len记录结束标记
			byte[] buf = new byte[1024];
			int len = -1;
			//新建字节数组输出流(内部维护了一个字节数组)
			baos = new ByteArrayOutputStream();
			//循环读取,写入baos中的字节数组
			while ((len = bos.read(buf)) != -1) {
				baos.write(buf, 0, len);
			}
			//获取baos中的字节数组
			byte[] bytes = baos.toByteArray();
			//将该字节数组变成字节码文件
			clazz = defineClass(null, bytes, 0, bytes.length);
		} catch (IOException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		//关闭流
		} finally {
			try {
				if (bos != null) {
					bos.close();
				}
			} catch (IOException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
			}
			try {
				if (baos != null) {
					baos.close();
				}
			} catch (IOException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
			}
		}
		//加载过程中失败则终止程序,否则返回该类
		if (clazz == null) {
			throw new RuntimeException("类不存在");
		} else {
			return clazz;
		}
		
	}
	/**
	 * 构造函数传递待加载类所在的目录
	 * @param classDir	待加载的类所在的目录
	 */
	public MyClassLoader(String classDir) {
		this.classDir = classDir;
	}

}

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值