节选自周志明的 《深入理解java虚拟机》
参考:
https://www.cnblogs.com/xing901022/p/5507086.html 父类加载过程,这个文章写的太好了,我直接copy了。
继承的加载顺序
由于static块会在首次加载类的时候执行,因此下面的例子就是用static块来测试类的加载顺序。
package xing.test.thinking.chap7;
class A{
static{
System.out.println("A static");
}
}
class B extends A{
static{
System.out.println("B static");
}
}
class C extends B{
private static D d = new D();
static{
System.out.println("C static");
}
}
class D{
static{
System.out.println("D static");
}
}
public class ExtendTest {
public static void main(String[] args) {
C c = new C();
}
}
在上面的例子中,类C继承B,B继承A,而C有依赖于D。因此当创建C的时候,会自动加载C继承的B和依赖的D,然后B又会加载继承的A。只有A加载完,才能顺利的加载B;BD加载完,才能加载C。这就是类的加载顺序了。
A static
B static
D static
C static
所有的变量初始化完,才会执行构造方法
在类的加载过程中,只有内部的变量创建完,才会去执行这个类的构造方法。
package xing.test.thinking.chap7;
class A2{
B2 b2 = new B2();
static{
System.out.println("A static");
}
public A2() {
System.out.println("A2()");
}
}
class B2{
C2 c2 = new C2();
D2 d2 = new D2();
static{
System.out.println("B static");
}
public B2() {
System.out.println("B2()");
}
}
class C2{
static{
System.out.println("C static");
}
public C2() {
System.out.println("C2()");
}
}
class D2{
static{
System.out.println("D static");
}
public D2() {
System.out.println("D2()");
}
}
public class VarTest {
public static void main(String[] args) {
A2 a2 = new A2();
}
}
在上面的例子中,A2里面有B2变量,B2则有C2D2变量。因此类的加载还是先读取到哪个,就执行相应的静态块。
当依赖的对象都定义完,才会执行构造方法:
A static
B static
C static
C2()
D static
D2()
B2()
A2()
静态成员与普通成员类的加载区别
在类的加载过程中,静态成员类的对象,会优先加载;而普通成员类的对象则是使用的时候才回去加载。
package xing.test.thinking.chap7;
class A3{
B3 b3 = new B3();
static C3 c4 = new C3();
static{
System.out.println("A3");
}
}
class B3{
static{
System.out.println("B3");
}
}
class C3{
static{
System.out.println("C3");
}
}
public class StaticTest {
public static void main(String[] args) {
A3 a3 = new A3();
}
}
输出:
C3
A3
B3
总结
第一点,所有的类都会优先加载基类
第二点,静态成员的初始化优先
第三点,成员初始化后,才会执行构造方法
第四点,静态成员的初始化与静态块的执行,发生在类加载的时候。
第四点,类对象的创建以及静态块的访问,都会触发类的加载。
补充类构造方法的顺序
看代码:
package xing.test.thinking.chap8;
class A{
public A() {
System.out.println("A");
}
}
class B extends A{
public B() {
System.out.println("B");
}
}
class C extends B {
private D d1 = new D("d1");
private D d2 = new D("d2");
public C() {
System.out.println("C");
}
}
class D {
public D(String str) {
System.out.println("D "+str);
}
}
public class ExtendTest {
public static void main(String[] args) {
C c = new C();
}
}
执行结果:
A
B
D d1
D d2
C
因此可以得出结论:
- 首先会调用基类的构造方法
- 其次,调用成员的构造方法
- 最后,调用自己的构造方法
类初始化的顺序和步骤:
加载,验证,准备,解析,初始化,使用,卸载。
在虚拟机里,比较两个类是否“相等”的条件是:这两个类是由同一个虚拟机加载,且两个类的全路径相同。
这里所谓的相等,包括代表类的class对象的equals方法,isAssignableFrom方法,isInstance方法返回的结果相等。
看一个例子:
同一个类由不同的classLoader加载,所以显示false。
这个也不错:
http://blog.youkuaiyun.com/boyupeng/article/details/47951037
http://blog.youkuaiyun.com/eff666/article/details/52203406