Java中自定义ClassLoader和ClassLoader的使用

什么是ClassLoader

当前环境JDK1.8、eclipse

ClassLoader简称类加载器,主要用于加载和校验编译后的Java文件(即:以.class结尾的文件);

有哪些类加载器(ClassLoader)

  1. AppClassLoader(应用类加载器)
  2. ExtClassLoader(扩展类加载器注意:JDK1.8后被修改为平台类加载器)
  3. BootstrapClassLoader(启动类加载器)

获取类加载器

通过简单的demo来得到类加载器

/**
 * @description User类主要用于测试当前的类加载器
 * @author hy
 * @date 2019-08-08
 */
public class User {
	public static void main(String[] args) {
		User user = new User();
		Class<? extends User> userClass = user.getClass();
		System.out.println(userClass.getClassLoader());
		System.out.println(userClass.getClassLoader().getParent());
		System.out.println(userClass.getClassLoader().getParent().getParent());
	}
}

结果显示:
sun.misc.Launcher$AppClassLoader@73d16e93

sun.misc.Launcher$ExtClassLoader@15db9742
null//这里涉及到底层所以返回null

类加载器的加载顺序

由于当前的类加载器使用双亲模式

  1. 首先加载系统类加载器,此时系统类加载器会判断当前类是否已近是当前系统已定的类,如果是加载系统类,不会初始化被加载的类,不存在则由ExtClassLoader加载
  2. ExtClassLoader检测加载,一般都是lib包中,不存在则交给AppClassLoader加载
  3. AppClassLoader检测加载类,当前应用加载器会从当前应用中(就是启动类或者整个程序中)查找需要加载的类,存在即加载程序的类,不存在交给用户定义的类加载器处理

使用双亲模式的好处,可以保护Java程序的安全,防止非法的加载Class

用户可以使用自定义类加载器用来实现对不同位置的类的加载和调用

自定义类加载器


public class MyClassLoader extends ClassLoader {
	/**
	 * @description 解析类文件获得当前解析后的类
	 * @param fileName 档期类的完全限定名
	 * @return 使用类加载器后获取的类
	 * @throws Exception 解析错误
	 */
	public Class<?> transClassFile(String fileName) throws Exception {
		byte[] classBytes = this.loadBinaryClassFile();
		//主要通过父类来解析当前的class二进制文件
		Class<?> clazz = super.defineClass(fileName, classBytes, 0, classBytes.length);
		return clazz;
	}

	/**
	 * @description 读取并加载类文件获得byte数组返回
	 * @return byte[] 数组
	 * @throws Exception 读取失败
	 */
	private byte[] loadBinaryClassFile() throws Exception {
		String classFilePath = "D:/Person.class";// 设置当前class文件的路径
		File classFile = new File(classFilePath);
		System.out.println("文件的长度:" + classFile.length());
		if (!classFile.exists()) {// 判断文件是否
			throw new FileNotFoundException("文件不存在。。。。。。。。。。");
		}
		InputStream fis = null;
		ByteArrayOutputStream bos = null;// 内存流
		byte[] bytes = new byte[(int) classFile.length()]; // 设置缓冲区

		byte[] readBytes = null;
		try {
			bos = new ByteArrayOutputStream();
			// 开始实例化流,并加载流
			fis = new FileInputStream(classFile);// 这里必须为文件的实际路劲
			while (( fis.read(bytes)) != -1) {
				bos.write(bytes);
			}
			readBytes = bos.toByteArray();
			System.out.println("byte数组的长度:" + readBytes.length);
		} catch (Exception e) {
			e.printStackTrace();
		} finally {
			if (fis != null) {
				fis.close();
			}
			if (bos != null) {
				bos.close();
			}
		}
		return readBytes;
	}
}

当前的Person中的内容为

package com.hy.classloader.demo;
public class Person{
	public Person(){
		System.out.println("Person 类被创建实例");
   }
}

测试自定义类加载器

将创建好的Person.java复制到D盘,使用cmd来编译Person.java并获得Person.class文件

编写测试类

package com.hy.classloader.demo;

public class MyClassLoaderTest {
	public static void main(String[] args) throws InstantiationException, IllegalAccessException {
		MyClassLoader myClassLoader = new MyClassLoader();
		try {
			Class<?> loadClass = myClassLoader.transClassFile("com.hy.classloader.demo.Person");
			Object newInstance = loadClass.newInstance();//实例化当前Class文件并创建当前文件的实例
			System.out.println(newInstance);
		} catch (Exception e) {
			System.out.println("使用自定义类加载器加载类失败。。。。");
			e.printStackTrace();
		}
	}
}

结果:Person 类被创建实例

总结:
1.使用自定义的类加载器的时候需要继承ClassLoader来实现class的加载
2.在加载的过程中须要使用 ByteArrayOutputStream 内存流
3.解析的时候需要使用父类来解析获得二进制信息以此得到Class的信息(必须调用 super.defineClass(fileName, classBytes, 0, classBytes.length);)
4.使用自定义的类加载器可以没有限制的在其他的地方加载类

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
JavaClassLoader是一个关键组件,它负责将Java类加载到JVMJavaClassLoader可以分为三个层次:Bootstrap ClassLoader、Extension ClassLoader和Application ClassLoader自定义ClassLoader可以使得我们更好地控制Java类的加载过程,例如可以从特定的路径或者网络加载类。 下面是一个简单的自定义ClassLoader的示例代码: ```java public class MyClassLoader extends ClassLoader { private String classPath; public MyClassLoader(String classPath) { this.classPath = classPath; } @Override protected Class<?> findClass(String name) throws ClassNotFoundException { byte[] data = loadClassData(name); return defineClass(name, data, 0, data.length); } private byte[] loadClassData(String name) { String fileName = classPath + File.separatorChar + name.replace('.', File.separatorChar) + ".class"; try { FileInputStream fis = new FileInputStream(fileName); ByteArrayOutputStream baos = new ByteArrayOutputStream(); int len; byte[] buffer = new byte[1024]; while ((len = fis.read(buffer)) != -1) { baos.write(buffer, 0, len); } return baos.toByteArray(); } catch (Exception e) { e.printStackTrace(); } return null; } } ``` 上述代码,我们继承了ClassLoader类,并实现了findClass方法,在该方法,我们可以根据自己的需求去加载Java类。在示例,我们从指定的路径加载类的字节码文件,并将其转换为字节数组,最后调用defineClass方法生成Class对象。注意,这里的路径需要与ClassLoader所在的类路径相对应。 我们可以通过以下代码来使用自定义ClassLoader: ```java MyClassLoader myClassLoader = new MyClassLoader("/path/to/class/files"); Class<?> clazz = myClassLoader.loadClass("com.example.Test"); Object obj = clazz.newInstance(); Method method = clazz.getMethod("hello"); method.invoke(obj); ``` 上述代码,我们通过自定义ClassLoader加载了Test类,并调用了hello方法。 需要注意的是,JavaClassLoader是一个层级结构,类的加载过程会从上至下依次进行,因此我们需要根据具体的需求来选择ClassLoader的层次。在自定义ClassLoader时,我们需要保证其所在的类路径与被加载的类所在的类路径相对应,否则就会出现ClassNotFoundException。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值