类的加载与初始化

类的加载与初始化

类的加载

(1)类加载的基本概念

与C或C++不同,Java程序并不是一个可执行文件,而是由许多独立的类文件组成的,每一个文件对应一个Java类。

这些类文件并不是一开始全部加载进内存,而是根据程序需要逐渐载入。ClassLoarder是JVM实现的一部分,ClassLoarder包括:

Bootstrap ClassLoarder(启动类加载器),在JVM运行的时候加载Java核心的API,以满足Java程序员最基本的需求。Extension ClassLoade和Application ClassLoader也在此时被加载。

Extension ClassLoade:加载Java的扩展API,也就是/lib/ext中的类。

Application ClassLoader:加载用户机器上CLASSPATH设置目录中的Class,在没有指定ClassLoader的情况下,程序员自定义的类就由该ClassLoader进行加载。

(2)类加载的基本流程

类加载的基本流程是:当运行一个程序的时候,JVM启动,运行Bootstrap ClassLoader,该ClassLoader加载Java核心API,然后调用Extension ClassLoade加载扩展API,最后Application ClassLoader加载CLASSPATH目录下定义的Class。

类加载过程中使用了一种父类委托模式,即如果一个类加载器接收到了类加载的请求,它首先把这个请求委托给他的父类加载器去完成,每个层次的类加载器都是如此,因此所有的加载请求都应该传送到顶层的启动类加载器中,只有当父加载器反馈自己无法完成这个加载请求(它在搜索范围中没有找到所需的类)时,子加载器才会尝试自己去加载。

在这里插入图片描述

使用父类委托模式的原因是可以避免重复加载,当父类已经加载了该类的时候,就没有必要子ClassLoader再加载一次。

(3)重要方法

loadClass方法:ClassLoader.loadClass()是ClassLoader的入口点。该方法的定义如下:

Class loadClass(String name,boolean resolve);

name是指JVM需要加载的类的名称,resolve参数告诉方法是否需要解析该类。在准备执行类之前,应该考虑类解析。注意:并不总是需要解析。如果JVM只是需要知道该类是否存在或找出该类的超类,那么就不需要解析。

defineClass方法:接受由原始字节组成的数组,并把它转换成Class对象。defineClass方法被标记成final,所以不能覆盖它。

findSystemClass方法:从本地文件系统中装入文件。它在本地文件系统中寻找类文件,如果存在,就使用defineClass方法将原始字节转换成Class对象,以将该文件转换成类。

resolveClass方法:解析类。

findLoadedClass方法:充当一个缓存,当请求loadClass装入类时,它调用该方法来查看ClassLoader是否已装入这个类。

forName方法:是Class类中的一个静态方法和ClassLoader中的loadClass方法的目的一样,都是用来加载class的,但是两者有所区别:loadClass加载类实际上就是加载的时候并不对该类进行解释,因此不会初始化该类。而forName方法加载类时会将类进行解释和初始化。

类的初始化

当我们第一次使用某一个类时,会先进行类加载,连接,验证,初始化,解释。类初始化是为类变量分配内存并设置默认值。

类变量是类中由static修饰的变量,它们在类初始化时在静态代码块中进行初始化。在编译的时候,编译器会自动收集类中的所有静态变量(类变量)和静态语句块(static{}块)中的语句合并产生的,编译器收集的顺序是根据语句在java代码中的顺序决定的。

收集完成之后,会编译成java类的 static{} 方法,java虚拟机则会保证一个类的static{} 方法在多线程或者单线程环境中正确的执行,并且只执行一次。在执行的过程中,便完成了类变量的初始化。值得说明的是,如果我们的java类中,没有显式声明static{}块,如果类中有静态变量,编译器会默认给我们生成一个static{}方法。

类的初始化与对象初始化

当我们第一次创建一个类的对象时,会先调用父类的静态代码块初始化该类的父类,再调用本类的静态代码块初始化本类。

而对象的初始化是在代码块和构造方法中进行的,所以接下来是调用父类的代码块和构造方法,再调用本类的代码块和构造方法。
例如:

public class Test {
	public static void main(String[] args) {
		System.out.println("---创建第一个B类对象---");
		B b = new B();
		System.out.println("---创建第二个B类对象---");
		B b2 = new B();
	}
}

class A {
	public A() {
		System.out.println("A构造方法");
	}
	
	{
		System.out.println("A代码块");
	}
	
	static {
		System.out.println("A静态代码块");
	}
}

class B extends A {
	public B() {
		System.out.println("B构造方法");
	}
	
	{
		System.out.println("B代码块");
	}
	
	static {
		System.out.println("B静态代码块");
	}
}

运行结果:
在这里插入图片描述

猜你喜欢

转载自blog.csdn.net/is_Javaer/article/details/82773920