类加载器

网友投稿 686 2022-05-30

类加载过程

加载->连接->初始化。连接过程又可分为三步:验证->准备->解析。

类加载器分类

JVM 中内置了三个重要的 ClassLoader,除了 BootstrapClassLoader 其他类加载器均由 Java 实现且全部继承自java.lang.ClassLoader:

启动类加载器(Bootstrap ClassLoader)此类加载器负责将存放在 \lib 目录中的,或者被 -Xbootclasspath 参数所指定的路径中的,并且是虚拟机识别的(仅按照文件名识别,如 rt.jar,名字不符合的类库即使放在 lib 目录中也不会被加载)类库加载到虚拟机内存中。启动类加载器无法被 Java 程序直接引用,用户在编写自定义类加载器时,如果需要把加载请求委派给启动类加载器,直接使用 null 代替即可。

扩展类加载器(Extension ClassLoader)这个类加载器是由 ExtClassLoader(sun.misc.Launcher$ExtClassLoader)实现的。它负责将 /lib/ext 或者被 java.ext.dir 系统变量所指定路径中的所有类库加载到内存中,开发者可以直接使用扩展类加载器。

应用程序类加载器(Application ClassLoader)这个类加载器是由 AppClassLoader(sun.misc.Launcher$AppClassLoader)实现的。由于这个类加载器是 ClassLoader 中的 getSystemClassLoader() 方法的返回值,因此一般称为系统类加载器。它负责加载用户类路径(ClassPath)上所指定的类库,开发者可以直接使用这个类加载器,如果应用程序中没有自定义过自己的类加载器,一般情况下这个就是程序中默认的类加载器。

双亲委派模型

每一个类都有一个对应它的类加载器。系统中的 ClassLoder 在协同工作的时候会默认使用 双亲委派模型 。

1、在类加载的时候,系统会首先判断当前类是否被加载过。已经被加载的类会直接返回,否则才会尝试加载。

2、加载的时候,首先会把该请求委派该父类加载器的 loadClass() 处理,因此所有的请求最终都应该传送到顶层的启动类加载器 BootstrapClassLoader 中。当父类加载器无法处理时,才由自己来处理。

3、当父类加载器为null时,会使用启动类加载器 BootstrapClassLoader 作为父类加载器。

好处

使得 Java 类随着它的类加载器一起具有一种带有优先级的层次关系,从而使得基础类得到统一。

例如 java.lang.Object 存放在 rt.jar 中,如果编写另外一个 java.lang.Object 并放到 ClassPath 中,程序可以编译通过。

由于双亲委派模型的存在,所以在 rt.jar 中的 Object 比在 ClassPath 中的 Object 优先级更高,这是因为 rt.jar 中的 Object 使用的是启动类加载器,而 ClassPath 中的 Object 使用的是应用程序类加载器。

rt.jar 中的 Object 优先级更高,那么程序中所有的 Object 都是这个 Object。

保证了Java程序的稳定运行,可以避免类的重复加载(JVM 区分不同类的方式不仅仅根据类名,相同的类文件被不同的类加载器加载产生的是两个不同的类),也保证了 Java 的核心 API 不被篡改。

如果没有使用双亲委派模型,而是每个类加载器加载自己的话就会出现一些问题,比如我们编写一个称为 java.lang.Object 类的话,那么程序运行的时候,系统就会出现多个不同的 Object 类。

注意

这个双亲翻译的容易让人误解,一般理解双亲都是父母,这里的双亲表达的是“父母一辈”的人,并不是说真的有一个 Mother ClassLoader 和一个 Father ClassLoader 。另外,类加载器之间的“父子”关系也不是通过继承来体现的,是由“优先级”来决定。官方API文档对这部分的描述如下:

The Java platform uses a delegation model for loading classes. The basic idea is that every class loader has a "parent" class loader. When loading a class, a class loader first "delegates" the search for the class to its parent class loader before attempting to find the class itself.

源码分析

private final ClassLoader parent;

protected Class loadClass(String name, boolean resolve)

throws ClassNotFoundException

{

synchronized (getClassLoadingLock(name)) {

// 首先检查请求的类是否已经被加载过

Class c = findLoadedClass(name);

if (c == null) {

long t0 = System.nanoTime();

try {

if (parent != null) {

//父加载器不为空,调用父加载器loadClass()方法处理

c = parent.loadClass(name, false);

} else {

//父加载器为空,使用启动类加载器 BootstrapClassLoader 加载

c = findBootstrapClassOrNull(name);

}

} catch (ClassNotFoundException e) {

//父类加载器无法完成加载请求

}

if (c == null) {

long t1 = System.nanoTime();

//自己尝试加载

c = findClass(name);

// this is the defining class loader; record the stats

sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);

sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);

sun.misc.PerfCounter.getFindClasses().increment();

}

}

if (resolve) {

resolveClass(c);

}

return c;

}

}

自定义加载器

除了 BootstrapClassLoader 其他类加载器均由 Java 实现且全部继承自java.lang.ClassLoader。如果我们要自定义自己的类加载器,很明显需要继承 ClassLoader。

示例:自定义一个NetworkClassLoader,用于加载网络上的class文件

package classloader;

import java.io.ByteArrayOutputStream;

import java.io.InputStream;

import java.net.URL;

类加载器

/**

* 加载网络class的ClassLoader

*/

public class NetworkClassLoader extends ClassLoader {

private String rootUrl;

public NetworkClassLoader(String rootUrl) {

this.rootUrl = rootUrl;

}

@Override

protected Class findClass(String name) throws ClassNotFoundException {

Class clazz = null;//this.findLoadedClass(name); // 父类已加载

//if (clazz == null) { //检查该类是否已被加载过

byte[] classData = getClassData(name); //根据类的二进制名称,获得该class文件的字节码数组

if (classData == null) {

throw new ClassNotFoundException();

}

clazz = defineClass(name, classData, 0, classData.length); //将class的字节码数组转换成Class类的实例

//}

return clazz;

}

private byte[] getClassData(String name) {

InputStream is = null;

try {

String path = classNameToPath(name);

URL url = new URL(path);

byte[] buff = new byte[1024*4];

int len = -1;

is = url.openStream();

ByteArrayOutputStream baos = new ByteArrayOutputStream();

while((len = is.read(buff)) != -1) {

baos.write(buff,0,len);

}

return baos.toByteArray();

} catch (Exception e) {

e.printStackTrace();

} finally {

if (is != null) {

try {

is.close();

} catch(IOException e) {

e.printStackTrace();

}

}

}

return null;

}

private String classNameToPath(String name) {

return rootUrl + "/" + name.replace(".", "/") + ".class";

}

}

JAR Java

版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:few-shot在关系抽取中的应用
下一篇:基于HiLens Studio开发行人检测与跟踪应用
相关文章