静态代码块、构造代码块和构造函数的执行顺序

静态代码块

static
{
  执行语句
}

在java中使用static关键字声明的代码块。静态块用于初始化类,为类的属性初始化。
每个静态代码块只会执行一次。由于JVM在加载类时会执行静态代码块,所以静态代码块先于主方法执行。
如果类中包含多个静态代码块,那么将按照"先定义的代码先执行,后定义的代码后执行"。
注意:1 静态代码块不能存在于任何方法体内。2 静态代码块不能直接访问静态实例变量和实例方法,需要通过类的实例对象来访问。

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

class Zi extends Fu {
    static {
        System.out.println("静态代码块Zi");
    }
    {
        System.out.println("构造代码块Zi");
    }
    public Zi() {
        System.out.println("构造方法Zi");
    }
}
public class demo {
	public static void main(String [] argus) throws ClassNotFoundException{
		Zi zi1 = new Zi();
		System.out.println("------------------");
		Zi zi2 = new Zi();
	}
}

输出:

静态代码块Fu
静态代码块Zi
构造代码块Fu
构造方法Fu
构造代码块Zi
构造方法Zi
------------------
构造代码块Fu
构造方法Fu
构造代码块Zi
构造方法Zi

执行顺序:main()->父类静态代码块->子类静态代码块>父类构造代码块->父类构造方法->子类构造代码块->子类的构造方法
分析:

1.jvm调用了main方法,main进栈;
2. 遇到Zi z = new Zi();会先将Fu.class和Zi.class分别加载进内存,再创建对象,当Fu.class加载进内存,父类的静态代码块会随着Fu.class一起加载,当Zi.class加载进内存,子类的静态代码块会随着Zi.class一起加载第一个输出,静态代码块Fu,第二个输出,静态代码块Zi;
3. 走Zi类的构造方法,因为java是分层初始化的,先初始化父类,再初始化子类,所以先走的父类构造函数,但是执行父类构造函数时,发现父类有构造代码块,构造代码块是优于构造函数执行的所以第三输出构造代码块Fu,第四输出构造方法Fu;
4. Fu类初始化结束,子类初始化,第五个输出的是构造代码块Zi,构造方法Zi。

猜你喜欢

转载自blog.csdn.net/u014253011/article/details/84564308