ClassLoad类加载器与双亲委派模型

1. 类加载器

Class类描述的是整个类的信息,在Class类中提供的方法getName()是根据ClassPath配置的路径来进行类加载的。若类加载的路径为文件、网络等时则必须进行类加载这是就需要用到ClassLoader类。

ClassPath:加载类的路径。

(1)ClassLoad

类加载器用来加载 Java类到Java 虚拟机中。Java 源程序(.java 文件)在经过 Java 编译器编译之后就被转换成Java字节代码(.class 文件)。类加载器负责读取 Java 字节代码,并转换成 java.lang.Class 类的一个实例

      

观察类加载器的存在性:

//自定义类,也在ClassPath路径下
class Message{}
public class Test2{
	public static void main(String[] args) throws Exception {
		Class<?> class1 = Class.forName("reflect.Message");
		System.out.println(class1.getClassLoader());
		System.out.println(class1.getClassLoader().getParent());
	System.out.println(class1.getClassLoader().getParent().getParent());
	}
}

      

通过上述程序结果我们可以看到两类类加载器:AppClassLoaderExtClassLoad;实则jdk默认提供如下几种类加载器:

A. BootStrap:

也称为启动类加载器或核心类加载器。由C++实现,是JVM虚拟机的一部分。

BootStrap外其他类加载器是由Java语言实现,独立于JVM并且继承与iava.lang.ClassLoader

主要负责加载JAVA_HOME/lib(典例:rt.jar

bootStrap类加载器不能直接被程序使用。

B. ExtClassLoader:

扩展类加载器,主要加载JAVA_HOME/lib/ext

Java语言实现可以被Java程序直接使用。

C. AppClassLoader:

应用程序类加载器,主要负责用户类路径(ClassPath)配置下的类库。若用户为自定义类加载器则默认使用AppClassLoader

以上三种类加载器加载的代码都必须要求在CLASSPATH中加载。

类加载器带来的好处:

可以通过动态的路径进行类的加载操作。

(1)自定义类加载器

用户决定类从哪里加载。

ClassLoad类中提供进行类加载的方法:

protected Class<?> loadClass(String name, boolean resolve)

比较两个类相等的前提:

必须是由同一个类加载器加载的前提下才有意义。否则,即使两个类来源于同一个Class 文件,被同一个虚拟机加载,只要加载他们的类加载器不同,那么这两个类注定相等。

2. 类加载器的双亲委派模型


(1)双亲委派模型定义

四中类加载器的层次关系就称为双亲委派模型。

(2)双亲委派模型中除了BootStrap外,其他类加载器都有自己的父类加载器。

(3)工作流程:

当一个类加载器收到加载类请求时,先不自己处理而是把加载请求委托给父加载器处理,只有当所有父加载器无法加载此类时子类才尝试自己加载。

好处:保证Java程序的稳定执行。

(4)双亲委派模型从jdk1.2之后引入,但它不强制约束,甚至可以破坏双亲委派模型来进行类加载(典型:OSGI技术:Java模块化技术)。

(5)双亲委派模型可保证Java程序的稳定执行(如:自定义javaLang包下的Object类,类加载器始终都不会对自定义的进行加载)。





猜你喜欢

转载自blog.csdn.net/qq_40409115/article/details/80285972