一.基本概念
a.包名的作用就是给类加载指明了class所在目录
b.类加载 就是把.class文件读进内存中 转换成一系列的可执行序列
c.将class文件加载到JVM的基本工作流程
(1) 装载:查找和导入Class文件;
(a) 通过一个类的全限定名来获取定义此类的二进制字节流
(b) 将这个字节流所代表的静态存储结构转化为方法区的运行时数据结构
(c) 在Java堆中生成一个代表这个类的java.lang.Class对象,作为方法区这些数据的访问入口。
虚拟机规范中并没有准确说明二进制字节流应该从哪里获取以及怎样获取,这里可以通过定义自己的类加载器去控制字节流的获取方式。
(2) 链接:把类的二进制数据合并到JRE中;
(a)校验:检查载入Class文件数据的正确性;
(b)准备:给类的静态变量分配存储空间;
(c)解析:将符号引用转成直接引用;
(3) 初始化:对类的静态变量,静态代码块执行初始化操作
d.双亲委派模型
一个类加载器收到了类加载的请求,它首先把这个请求委托给他的父类加载器去完成,父类加载不了的再交由子类加载。
优点:所有的类只会被一个类加载器加载。避免了重复加载这中混乱的情况出现。
e.类加载器继承关系
(1) Bootstrap ClassLoader : 将存放于<JAVA_HOME>\lib目录中的,或者被-Xbootclasspath参数所指定的路径中的,并且是虚拟机识别的
(仅按照文件名识别,如 rt.jar 名字不符合的类库即使放在lib目录中也不会被加载)类库加载到虚拟机内存中。启动类加载器无法被Java程序直接引
用
(2) Extension ClassLoader : 将<JAVA_HOME>\lib\ext目录下的,或者被java.ext.dirs系统变量所指定的路径中的所有类库加载。开发者可以
直接使用扩展类加载器。
(3) Application ClassLoader : 负责加载用户类路径(ClassPath)上所指定的类库,开发者可直接使用。
(4)这一层就是自定的类加载器喽.
参考链接 https://www.cnblogs.com/ITtangtang/p/3978102.html
二.自定义类加载demo例子
1.定义一个从本地磁盘加载class的 DiskClassLoader
package com.zhao.test.loader;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.InputStream;
public class DiskClassLoader extends ClassLoader {
protected Class<?> findClass(String classname) throws ClassNotFoundException {
byte[] classByte = getClassByte(classname);
if (classByte == null) {
throw new ClassNotFoundException();
}
return defineClass(classname, classByte, 0, classByte.length);
}
//将本地的class文件读进内存当中
public byte[] getClassByte(String classname) {
File file = new File(findClassPath(classname));
InputStream inputStream = null;
try {
ByteArrayOutputStream outputStream = new ByteArrayOutputStream();
inputStream = new FileInputStream(file);
byte b[] = new byte[8000];
int flag = 0;
while ((flag = inputStream.read(b)) != -1) {
outputStream.write(b, 0, flag);
}
return outputStream.toByteArray();
} catch (Exception e) {
e.printStackTrace();
}
return null;
}
public String findClassPath(String classname) {
return "/" + classname.replace(".", "/")+".class";
}
}
2.本地写一个待加载的Class 编译成.class文件
//包 就是一个目录结构
package home.zhao;
public class LoadMain {
public String print(){
System.out.println("执行自定义方法");
return "这是一个返回值";
}
}
3.启动类测试
package com.zhao.test.loader;
public class TestMain {
public static void main(String args[]) {
DiskClassLoader loader = new DiskClassLoader();
try {
Class<?> haHa = loader.findClass("home.zhao.LoadMain");
Object o = haHa.newInstance();
System.out.println(haHa.getDeclaredMethods()[0].invoke(o));
} catch (Exception e) {
e.printStackTrace();
}
}
}