ClassLoader 简介
ClassLoader 在 Java 中有着非常重要的作用,它主要工作在 Class 装载的加载阶段。
其主要作用是从系统外部获得 Class 二进制数据流。
它是 Java 的核心组件,所有的 Class 都是由 ClassLoader 加载的,ClassLoader 负责通过将 Class 文件里的二进制数据流装载进系统,然后交给 Java 虚拟机进行连接、初始化等操作。
ClassLoader 种类
有如下四种 ClassLoader:
- BootStrap ClassLoader(启动类加载器):C++ 编写,加载核心库 java.*
- Extension ClassLoader(扩展类加载器):Java 编写,加载扩展库 javax.*
- Application ClassLoader(应用程序类加载器):Java 编写,加载程序所在目录(比如我们项目工程里的 src 目录)
- 自定义类加载器:Java 编写,定制化加载
本文就是动手写一个简单的第四种类加载器:自定义类加载器。
准备工作
实现自己的类加载器需要继承 ClassLoader
我们可以看到这是一个抽象类,但是你会发现整个类中一个抽象方法都没有。。。
ClassLoader 类中方法琳琅满目,但是对于我们要实现自己的 ClassLoader 来说,我们只需要关注以下这两个方法。
findClass(String name)
:顾名思义,就是根据路径名 name ,去找到对应的 class 文件。defineClass(byte[] b, int off, int len)
:顾名思义,定义一个类,把传进来的字节流转化为 Class 对象。
java.lang.ClassLoader 的 loadClass() 实现了双亲委派模型的逻辑,自定义类加载器一般不去重写它,但是需要重写 findClass() 方法。我们重写 findClass() 方法即可。
实现自己的 ClassLoader
首先,我们先在自己的桌面用文本编辑器写一个简单的类文件,如下所示
public class Person {
static {
System.out.println("I love you three thousand");
}
}
我们知道,静态代码块会在类执行初始化(这个初始化不是指 new 对象实例的初始化,而是指类装载过程中的最后一步)的时候执行,我们一会儿就根据这个特点来验证我们的类加载器是否正确的将类加载进了 Java 虚拟机。
之后实现我们自己的 ClassLoader ,非常简单,大致的思想我都写在注释里面了,这里不多说了。
package com.util.xgb;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
public class MyClassLoader extends ClassLoader{
private String path;// 路径
private String classLoaderName;// 类加载器名称
public MyClassLoader(String path, String classLoaderName) {
this.path = path;
this.classLoaderName = classLoaderName;
}
//用于寻找类文件
@Override
protected Class findClass(String name) throws ClassNotFoundException {
byte[] b = loadClassData(name);
// 直接调用 defineClass 来获取 Class 对象
return defineClass(name, b, 0, b.length - 1);
}
//用于加载类文件,说白了就是将 class 文件转换成字节数组
private byte[] loadClassData(String name) {
//类文件的全路径
name = path + name + ".class";
InputStream in = null;
ByteArrayOutputStream out = null;
try {
//读取字节码文件中的字节流
in = new FileInputStream(new File(name));
out = new ByteArrayOutputStream();
int i = 0;
while((i = in.read()) != -1) {
out.write(i);
}
} catch (Exception e) {
e.printStackTrace();
} finally {
// 保证资源的释放
try {
out.close();
in.close();
} catch (IOException e) {
e.printStackTrace();
}
}
return out.toByteArray();
}
}
然后,我们写一个测试类简单的测试一下看看
package com.util.xgb;
public class MyClassLoaderTest {
public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException {
// 初始化我们自己的类加载器,记得使用你刚刚写的 Person 文件的全路径
MyClassLoader m =
new MyClassLoader("C:\\Users\\_Miracle\\Desktop\\", "myClassLoader");
Class c = m.findClass("Person");
System.out.println(c.getClassLoader());
c.newInstance();
}
}
执行代码
gg,找不到文件,原因是我们没有编译刚写的 Person.java 文件,使用 javac 指令编译一下。
再执行一次,看看打印结果
com.util.xgb.MyClassLoader@5e2dc80c
I love you three thousand
完成。