一文详解:Java中父子类静态块构造块构造方法成员变量之间的初始化先后顺序与执行先后顺序

Posted 睡竹

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了一文详解:Java中父子类静态块构造块构造方法成员变量之间的初始化先后顺序与执行先后顺序相关的知识,希望对你有一定的参考价值。

1、各个块之间的执行顺序

父类静态代码块--->子类静态代码块
--->父类构造代码块--->父类构造方法
--->子类构造代码块--->子类构造方法

2、各个模块之间的加载顺序

加载顺序与执行顺序不同,这里指的是JVM加载类中的组成部分

父类静态成员变量/静态代码块/静态成员方法--->子类静态成员变量/静态代码块/静态成员方法
--->父类普通成员变量/普通成员方法--->父类构造代码块--->父类构造方法
--->子类普通成员变量/普通成员方法--->子类构造代码块--->子类构造方法

注意:

  1. 静态成员变量、静态成员方法静态代码块属于同一级别加载,根据代码书写的先后顺序,决定谁先加载。

  1. 静态代码块若使用了某个静态成员变量,那么该静态成员变量一定比静态代码块先加载。

下面为该顺序的详细解释

2、释义

  • 在Java中,由花括号" "包裹的范围,称为块

静态代码块
  • 最外层由static引领的块"",称为静态代码块,

特点:在类加载时(可以理解为程序启动时),自动执行一次,之后不会再执行

如:

public class ParentCase 
    /** 静态代码块 */
    static 
        System.out.println("我是静态代码块");
    
构造代码块
  • 在类中直接使用的块,称为构造代码块,又称为动态代理块或实例代码块

特点:每次创建对象(new)时,该块自动执行一次,用于初始化类中的配置,若该类没有创建对象,则构造代码块不会执行(而静态代码块在项目启动时,自动执行一次)

如:

public class ParentCase 
    /** 构造代码块 */
    
        //用于初始化该类的配置,如每次创建不同对象时,加载公共配置
        System.out.println("我是构造代码块");
    
构造方法
  • 方法名称与类名相同,无返回类型且不能使用static修饰的方法,称为构造方法。

特点:

  1. 在该类实例化时,会执行指定的构造方法去创建对象。

  1. 构造方法在构造代码块后执行(与构造代码块的区别)

  1. 每一个类都默认存在无参的构造方法,当手动声明构造方法时,无参的构造方法会消失。

如:

public class ParentCase 
    public ParentCase() 
    
成员变量
  • 直接声明在类中的变量。根据是否有static修饰,分为"静态成员变量"与"普通成员变量(又称为实例成员变量)"

注意:

  1. 类初始化时,遇到static模块(静态成员变量、静态代码块),会从上往下编译。

  1. 静态成员变量由static修饰,与静态代码块加载顺序是同一级别的,但是,若静态代码块中使用了某个成员变量,那么该成员变量必须声明在静态代码块之前(这时,静态成员变量比静态代码块先加载

如:

3、案例

案例:

父类代码

如下:

public class ParentCase 

    static 
        System.out.println("父类静态代码块");
    

    
        System.out.println("父类构造代码块");
    

    public ParentCase() 
        System.out.println("父类构造方法");
    
子类代码

如下:

public class ChildrenCase extends ParentCase

    static 
        System.out.println("子类静态代码块");
    

    
        System.out.println("子类构造代码块");
    

    public ChildrenCase() 
        System.out.println("子类构造方法");
    
测试代码

如下:

public static void main(String[] args) 
    /**
     * 创建子类对象,调用子类的成员方法test()
     * 测试各个块之间的执行顺序
     */
    new ChildrenCase();
结果

Java类的各种成员初始化顺序如:父子类继承时的静态代码块,普通代码块,静态方法,构造方法,等先后顺

class B extends A ,然后A类也就是父类里面有静态代码块,普通代码块,静态方法,静态成员变量,普通成员变量,普通方法。
子类也是这样,然后继承之后,关于程序打印输出的结果。
涉及到Java类的各种成员的初始化顺序。
经测试,得到如下结论:

1.父类【静态成员】和【静态代码块】,按在代码中出现的顺序依次执行。
2.子类【静态成员】和【静态代码块】,按在代码中出现的顺序依次执行。
3.父类的【普通成员变量被普通成员方法赋值】和【普通代码块】,按在代码中出现的顺序依次执行。
4.执行父类的构造方法。
5.子类的【普通成员变量被普通成员方法赋值】和【普通代码块】,按在代码中出现的顺序依次执行。
6.执行子类的构造方法。


下面看测试代码以及测试结果的图:

父类:

package com.lxk.ClassInitTest;/** * 父类 * <p> * Created by lxk on 2017/4/20 */public class Parent {static {System.out.println("父类:静态代码块");}{System.out.println("父类:普通代码块");}private static String staticStringInParent = initStaticStringInParent();private String stringInParent = initStringInParent();public Parent() {System.out.println("父类:构造方法");}private static String initStaticStringInParent() {System.out.println("父类:静态方法,被静态成员变量赋值调用。");return "initStaticStringInParent";}private String initStringInParent() {System.out.println("父类:普通成员方法,被普通成员变量赋值调用。");return "initStringInParent";}}

子类:

package com.lxk.ClassInitTest;/** * 子类 * <p> * Created by lxk on 2017/4/20 */public class Child extends Parent {private String stringInChild = initStringInChild();private static String staticStringInChild = initStaticStringInChild();{System.out.println("子类:普通代码块");}static {System.out.println("子类:静态代码块");}public Child() {System.out.println("子类:构造方法");}private static String initStaticStringInChild() {System.out.println("子类:静态方法,被静态成员变量赋值调用。");return "initStaticStringInChild";}private String initStringInChild() {System.out.println("子类:普通成员方法,被普通成员变量赋值调用。");return "initStringInChild";}}

测试main方法

package com.lxk.ClassInitTest;/** * 测试Java类的成员和初始化块(代码块)初始化顺序 * <p> * Created by lxk on 2017/4/20 */public class ClassInitTest {public static void main(String[] args) {System.out.println("测试代码开始");new Child();System.out.println("测试代码结束");}}


代码的执行结果图:

为了看的方便,我把代码执行结果,和得到的理论放在一个图。

技术分享图片


分析:

我在子类里面,调换了代码的 顺序,父类和子类的顺序是不一样的,看的时候注意,

这个也是测试结论的一环。

(2018.01.23 更新)

上面的描述有问题,

{System.out.println("父类:构造代码块");}

这东西,应该叫,构造代码块,构造 code block,不是我上面写的

下面再多科普一下吧
普通代码块:
方法或语句中出现的{}就称为普通代码块。
普通代码块和一般的语句执行顺序由他们在代码中出现的次序决定--“先出现先执行”
静态代码块:
在java中使用static关键字声明的代码块。
静态块用于初始化类,为类的属性初始化。每个静态代码块只会执行一次。
由于JVM在加载类时会执行静态代码块,所以静态代码块先于主方法执行。
如果类中包含多个静态代码块,那么将按照"先定义的代码先执行,后定义的代码后执行"。
注意:
1 静态代码块不能存在于任何方法体内。
2 静态代码块不能直接访问静态实例变量和实例方法,需要通过类的实例对象来访问。
构造代码块:
直接在类中定义且没有加static关键字的代码块称为{}构造代码块
构造代码块在创建对象时被调用,每次创建对象都会被调用,并且构造代码块的执行次序优先于类构造函数
这个构造代码块的执行顺序不会因为方法所在位置而影响,我特意将他放在构造函数之后。



以上是关于一文详解:Java中父子类静态块构造块构造方法成员变量之间的初始化先后顺序与执行先后顺序的主要内容,如果未能解决你的问题,请参考以下文章

java面试基础题------》Java 中的父子类静态代码块,代码块,构造方法执行顺序

java 父子类加载顺序

java中成员变量代码块构造函数运行顺序

java 对象的初始化流程(静态成员静态代码块普通代码块构造方法)

Java代码块

Java类加载