类加载是指将Java类的字节码从外部存储到内存中,并在JVM中创建对应的class对象的过程。
类加载的作用:
Java类加载过程可以分为以下三个阶段:
1.装载(Loading):将类的二进制数据读入内存,并为之创建一个java.lang.Class对象。通常,这个过程会从本地文件系统、网络或其他来源中获取类的字节码文件。
2.连接(Linking):分为验证、准备和解析三个阶段。
3.初始化(Initialization):为类的静态变量赋予正确的初始值,并执行类的初始化代码块。
Java里有如下几种类加载器
如下图,加载某个类的时候会先委托父类加载器寻找目标类,找不到在委托上层父类加载器寻找,如果都找不到目标类,则交给自己的路径的类加载器加载目标类。
简要的说:先找父类加载,不行再找子类加载。
自定义类加载器需要继承java.lang.ClassLoader类,该类有两个核心的方法:loadClass和findClass。
loadClass实现了双亲委派;findClass可以自定义类加载。
自定义类加载器步骤:
public class MyClassLoader extends ClassLoader {
private String classpath;
public MyClassLoader(String classpath) {
this.classpath = classpath;
}
@Override
protected Class<?> findClass(String name) throws ClassNotFoundException {
try {
byte[] bytes = loadByte(name);
return defineClass(name, bytes, 0, bytes.length);
} catch (IOException e) {
e.printStackTrace();
throw new ClassNotFoundException();
}
}
private byte[] loadByte(String className) throws IOException {
FileInputStream fis = new FileInputStream(classpath + File.separator + className.replace(".", File.separator).concat(".class"));
byte[] bytes = new byte[fis.available()];
fis.read(bytes);
fis.close();
return bytes;
}
}
在桌面上新建一个demo文件夹,并在文件夹内创建一个Test.java文件,内容如下:
public class Test {
public static void say() {
System.out.println("this is a static method!");
}
public void print(String s) {
System.out.println("printing:"+s);
}
}
使用javac命令编译成字节码文件,然后回到我们的开发工具测试:
public static void main(String[] args) throws Exception {
MyClassLoader myClassLoader = new MyClassLoader("C:\\Users\188\\Desktop\\demo");
Class<?> aClass = myClassLoader.loadClass("Test");
//调用的静态方法
aClass.getDeclaredMethod("say").invoke(aClass);
Object o = aClass.newInstance();
Method print = aClass.getDeclaredMethod("print", String.class);
print.invoke(o, "调用的对象方法");
System.out.println(aClass.getClassLoader());
System.out.println(aClass.getClassLoader().getParent());
System.out.println(aClass.getClassLoader().getParent().getParent());
System.out.println(aClass.getClassLoader().getParent().getParent().getParent());
}
Tomcat类加载打破双亲委派原因?