直接上例子
package com;
class Xhy{
public Xhy(){
System.out.println("Xhy的构造方法执行了");
}
}public class Pet extends Xhy{
private static int i = 9;
protected int j ;
public Pet(){
System.out.println("Pet的构造方法执行了");
System.out.println("i="+i+",j="+j);
j=39;
}
private static int x1 =printInit("static x1 执行了");
public static int printInit(String s){
System.out.println(s);
return 47;
}
}
class Dog extends Pet{
private int k = printInit("Dog 执行了");
public Dog(){
System.out.println("Dog的构造方法执行了");
System.out.println("k="+k);
System.out.println("j="+j);
}
private static int x2= printInit("static x2 执行了");
public static void main(String[] args) {
System.out.println("开始创建Dog对象");
Dog dog = new Dog();
System.out.println("对象创建完成");
}
}
执行流程
1.在Dog运行java时,第一件事就是访问Dog.main() (一个static方法),
2.于是加载器开始启动并找出Dog类的编译代码,在对它进行加载的过程中,发现Dog有一个父类(这是由关键字extends得知的),于是它继续进行加载,不管你是否打算产生一个Dog对象这时都会执行。(可以把我代码copy过去自己试验)。
3.如果该父类还有父类,那第二个父类也会被加载,以此类推。
4.必要的类加载完毕后,会从上往下依次开始找带有static修饰的代码并执行。
5.全部执行完毕后,则进入main方法执行。
6. 这时会创建一个Dog对象,在创建Dog对象的时候,首先,对象中的所有基本类型都会被设为默认值,对象引用被设置为null。
7. 然后创建对象的时候发现对象上有父类,则会加载父类,以此类推。直到全部加载完毕后。会依次从上往下执行构造方法。全部加载完毕后,对象创建完成。
这里就衍生出一个问题:为什么主方法不能输出实例变量
因为主方法是static修饰的,属于类级别的,而实例变量是属于对象级别的。主方法去访问实例变量的时候。这个时候主方法被调用了,但是实例变量因为不是static修饰的,还没有被创建出来,所以无法访问。
问题总结:
类级别比对象级别优先级要高。
序号不代表执行流程顺序,只是为了排版
执行结果