之前对于静态代码块,构造代码块,普通代码块,以及类的初始化等代码执行顺序一直不是特别清楚,在此整理,方便复习!
执行顺序:
- 父类 -->子类
- 静态属性&静态代码块 -->普通属性&构造代码块-->构造方法 -->普通代码块(所在方法被调用才执行)
原因分析:
静态代码块:
- 在java中使用static关键字声明的代码块{}。
static { System.out.println("静态代码块"); }
- 代码级别:成员级别
- ==在类加载时执行,且只执行一次==
(对于类加载不是很清楚的小伙伴们可先看这篇博客:类加载)class Parent { static { System.out.println("A"); } Parent(){ System.out.println("sdfsaf"); } static { System.out.println("B"); } } public class ExplicitStatic { public static void main(String[] args) { new Parent();//第一次执行,触发了类加载 System.out.println("=========华丽的分割线========="); new Parent();//第二次实例化,不再触发类的加载,不再执行静态代码块 } }
运行结果:
构造代码块:
- 直接在类中定义且没有加static关键字的代码块称为{}构造代码块。
public class Test01 { { System.out.println("我是构造代码块"); } public static void main(String[] args) { new Test01(); } }
- 代码级别:成员级别
- 构造代码块在创建对象时被调用,每次创建对象都会被调用
- 构造代码块的执行次序优先于类构造方法
class Test02{ { System.out.println("我是 父类 构造代码块"); } } public class Test01 extends Test02{ { System.out.println("我是 子类 构造代码块1"); } Test01(){ System.out.println("我是无参构造方法"); } Test01(int v){ System.out.println("我是有参构造方法"); } static { System.out.println("静态代码块"); } { System.out.println("我是 子类 构造代码块2"); } public static void main(String[] args) { new Test01(); System.out.println("===================华丽的分割线==================="); new Test01(6);//每次创建对象都会执行构造代码块 } }
运行结果:
普通代码块:
- 在方法或语句中出现的{}就称为普通代码块。
public void method() { { System.out.println("普通代码块1"); } .... }
- 代码级别:方法级别
- 普通代码块和一般的语句执行顺序由他们在代码中出现的次序决定--“先出现先执行”
class Test02{ { System.out.println("我是 父类 构造代码块"); } } public class Test01 extends Test02{ { System.out.println("我是 子类 构造代码块"); } static { System.out.println("静态代码块"); } public void method(){ { System.out.println("普通代码块1"); } System.out.println("普通方法体"); { System.out.println("普通代码块2"); } } public static void main(String[] args) { new Test01(); System.out.println("===================华丽的分割线==================="); new Test01().method();//被调用才会执行方法中的代码块 } }
运行结果:
类的实例化:
- 类的实例化有三部分:属性,构造代码块,构造方法
- 属性的加载和构造代码块加载是同等级的,执行顺序为书写顺序
- 构造方法在前两种方式加载完之后再执行
总结:
对象实例化时的顺序:
1,父类的静态成员变量和静态代码块加载
2,子类的静态成员变量和静态代码块加载
3,父类成员变量和构造代码块加载
4,父类的构造方法加载
5,子类成员变量和构造代码块加载
6,子类的构造方法加载
知识点汇总 ,测试源码如下:
class Parent2 { { System.out.println("父类 构造代码块1");//5 } Parent2() { System.out.println("父类 的无参构造方法");//7 } static { System.out.println("父类 的静态代码块1");//1 } { System.out.println("父类 构造代码块2");//6 } static { System.out.println("父类 的静态代码块2");//2 } } public class ExplicitStatic extends Parent2 { static { System.out.println("子类 的静态代码块1");//3 } ExplicitStatic() {//15 super();//可省略 System.out.println("子类 无参构造方法");//16 } ExplicitStatic(int v) {//14 this();//即 执行15 System.out.println("子类 有参构造方法");//17 } private int initA() { System.out.println("子类 的属性初始化调用 普通方法A");//9 return 0; } static { System.out.println("子类 的静态代码块2");//4 } private int initB() { System.out.println("子类 的属性初始化调用 普通方法B");//12 return 0; } int a = initA();//8 { System.out.println("子类 的构造代码块1");//10 } int b = initB();//11 { System.out.println("子类 的构造代码块2");//13 } public void method() { { System.out.println("子类 的普通代码块1"); } System.out.println("子类 的普通方法method"); { System.out.println("子类 的普通代码块2"); } } public static void main(String[] args) { new ExplicitStatic(100); System.out.println("================="); new ExplicitStatic().method(); }
运行结果如图:
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。