ClassLoader在Java中有着非常重要的作用,它主要工作在Class装载的加载阶段,其主要作用是从系统外部获得Class二进制数据流。它是Java的核心组件,所有的CLass都是由ClassLoader进行加载的,ClassLoader负责通过将Class文件里的二进制数据流程装载进系统,然后交给Java虚拟机进行连接、初始化等操作。
ClassLoader的种类有四类,分别是BootStrapClassLoader,ExtClassLoader,AppClassLoader,自定义CLassLoader
- BootStrapClassLoader:C++编写,加载核心库 java.*
- ExtClassLoader:Java编写,加载扩展库 javax.*
- AppClassLoader:Java编写,架子啊程序所在的目录
- 自定义CLassLoader:Java编写,定制化加载
今天,我们就来写一个自定义ClassLoader,加深对ClassLoader的学习。
首先,创建一个Java文件,放在D盘的根路径下,并用javac命令直接编译成class文件,这个java文件我是随便写的一个打印语句的,如下:
public class Wal
{
static{
System.out.println("Hello Wal");
}
}
然后再Idea中创建一个Java文件,MyClassLoader.java,这个就是我们的自定义ClassLoader了,代码如下:
package com.wal.javabasic.reflect;
import java.io.*;
/**
* 自定义ClassLoader
* @author wal
* @date 2019/6/28
*/
public class MyClassLoader extends ClassLoader{
private String path;
private String classLoaderName;
public MyClassLoader(String path, String classLoaderName) {
this.path = path;
this.classLoaderName = classLoaderName;
}
//用于寻找类文件
@Override
public Class findClass(String name){
byte[] b = loadClassData(name);
return defineClass(name, b, 0, b.length);
}
//用于加载类文件
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();
}
}
然后创建一个执行的类ClassLoaderChecker.java,去检验我们的自定义ClassLoader,代码如下:
package com.wal.javabasic.reflect;
/**
* 检验自定义ClassLoader
* @author wal
* @date 2019/6/28
*/
public class ClassLoaderChecker {
public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException {
MyClassLoader m = new MyClassLoader("D:\\", "myClassLoader");
Class c = m.loadClass("Wal");
System.out.println(c.getClassLoader());
c.newInstance();
}
}
最终的执行结果如下: