ClassLoader中三个阶段:装载 , 链接 , 初始化
装载即找到对应的.class文件加载进去jvm
链接的时候有三个步骤:验证,准备,解析三个小步骤 验证是为了确保这个.class文件在结构来说是符合java语言的语法结构的。
准备阶段的时候,会为该.class文件的静态资源分配存储空间,并为起赋予默认值。 最后阶段是解析 是将所有的符号引用转换为真实的引用,
在这一步是会对该.class文件引用的外部文件做加载操作的。一般的classloader 并不会执行这一步 调用的一般是loadClass(String,false)而不是load(String,true) 这样只有到真正用到该引用的时候才会去做其他外部.class文件的加载工作。
以上两段红字可以通过以下两段代码来进行验证:
下面这段代码中 我们讲TestClone的.class文件删除了,然后执行下面这段代码
public
class
TestRelationShip {
private
static
TestClone
test
=
new
TestClone(1);
public
void
print(){
System.
out
.println(
"i can run"
);
}
public
static
void
main(String[] args) {
System.
out
.println(
"main run"
);
}
}
执行结果是报错:由此可见 若存在static关键字 则在链接的准备阶段时 jvm回去查找TestClone的.class文件(这种情况在如果某个方法的参数为TestClone的引用的时候也会出现,可见这阶段已经在解析.class文件中的方法了,我是这么认为的)
下面我们去掉static关键字再看一下
public
class
TestRelationShip {
private
TestClone
test
=
new
TestClone(1);
public
void
print(){
System.
out
.println(
"i can run"
);
}
public
static
void
main(String[] args) {
System.
out
.println(
"main run"
);
}
}
执行结果通过: 由此可见 若不存在关键字 则在链接的准备价阶段 jvm并未去查找TestClone.class文件 事实上 jvm把这一步骤放到了解析的步骤中 但是classLoader一般都处在懒加载的状态 并不会去执行解析的步骤 所以main方法就成功执行了
最后即是初始化的过程 这个过程会去为所有属性赋予定义的值(注意:这里是你代码中给定义的值 不是jvm的默认值),并执行静态初始化块。
需要强调的是 初始化的过程并不是都走的 只有在一下几种条件下才会触发:
1.new 实例化对象
2.调用这个.class文件中的静态方法;
3.调用.class文件中的静态属性(这个属性不被final关键字修饰)
4.为静态属性赋值
5.在顶层java类中执行assert语句?(这个是抄来的。。不大理解。。)
可以这样理解:不管怎么折腾,你无法访问到一个没定义的值,jvm总会先你一步查看是否已经初始化,是否有初始化的必要(有的时候访问的是final和static修饰的属性,并且这个属性已经在链接阶段就赋上值了,这个时候就没必要初始化),如果有必要jvm就会初始化为你赋上值(当然了,如果你初始化的时候还是没有值那就只能取到链接的准备阶段时候,jvm为它附上的默认值了 (0,0.0 ,null))