本文为原创文章,转载请注明出处
查看[Java]系列内容请点击:https://www.jianshu.com/nb/45938443
本篇文章假设类已经编译完成,重点将jvm启动后的加载与实例化过程。
首先来整体看下我们new
一个对象都发生了什么:
当通过Class.forName("your class path")
,实际上是执行到第二步:寻找.class
文件并生成其对应的Class
对象,后续实例化的时候直接在堆上分配内存和进行构造方法的构造即可。
Class对象
Class
对象是用来描述类的对象,Class
对象中保存了类的各种信息,包括类的属性和字段信息,所以在使用反射机制的时候,经常需要从Class
对象中获取信息。对于一个类只会生成一个Class
对象。所以Class
对象可以用==
来进行比较。
在类内可以通过this.getClass()
获取Class
对象,也可以通过对象的.getClass()
方法获得。
枚举类型
需要特别说明的是,对于枚举类型,实际上枚举类型中的每一个值都是一个对象,枚举类型只能由JVM初始化,并且每个值仅有一个对象,所以枚举类型的值也可以用==
来比较。
由于枚举类型的对象只能由JVM初始化,不允许外界调用,所以枚举类型的构造函数只能是private
类型。
静态代码块和静态变量
前面我们说了,静态的代码块和静态变量是在类加载的时候执行的,也就是生成Class
对象的时候执行的。静态代码块如下:
static {
// your code here
}
而有一个例外,我们前面说了,枚举类型是由JVM在加载类的时候实例化的,同时枚举类型的静态代码块是在枚举实例构造之后运行的,与普通的类恰恰相反。
普通代码块和普通变量
普通代码块和普通变量是在对象初始化的时候运行的,也就是在new
一个对象的时候,生成Class
对象之后运行的。普通代码块的运行优先级高于构造方法,也就是说普通代码块是在构造方法之前运行的。
试一试
运行以下代码:
public enum EnumTest {
A, B, C;
static {
System.out.println("EnumTest loaded.");
}
{
System.out.println("11");
}
private EnumTest() {
System.out.println("EnumTest Constructor");
}
}
public class MainTest {
public static void main(String[] args) throws Exception {
Class.forName("EnumTest");
}
}
运行MainTest
的结果:
11
EnumTest Constructor
11
EnumTest Constructor
11
EnumTest Constructor
EnumTest loaded.
由于构造了3个枚举对象,所以运行了三次枚举类型的普通代码块和构造方法,而由于枚举类型是由JVM实例化的,所以这里的静态代码块在构造之后执行。
网友评论