静态代码块、非静态代码块、构造函数的输出顺序

时间:2021-09-23 19:29:15

代码示例:

情况一:没有继承父类时

class HelloA {
    
    public HelloA() {
        System.out.println("I'm A class");
    }

    static {
        System.out.println("static A");//特点:随着类的加载而执行,且只执行一次,并优先于主函数。用于给类初始化的。
    }
    
    {
        System.out.println("A");
    }

    public static void main(String[] args) {
        new HelloA();
        new HelloA();
    }
}

输出结果:

  static A
  A
  I'm A class
  A
  I'm A class

分析:输出顺序为: 静态代码块>非静态代码块>构造器,同时无论New多少次class静态代码块只执行一次,所以一般情况把耗内存重量级的连接或者其他放在静态代码块中,访问速度快,并且只访问一次,节约消耗。

情况二:继承某个父类时

class HelloA extends HelloB{
    
    public HelloA() {
        System.out.println("I'm A class");
    }

    static {
        System.out.println("static A");
    }
    
    {
        System.out.println("A");
    }

    public static void main(String[] args) {
        new HelloA();
    }
}

class HelloB {
    public HelloB() {
        System.out.println("I'm B class");
    }

    static {
        System.out.println("static B");
    }

    {
        System.out.println("B");
    }
}

运行结果:

  static B
  static A
  B
  I'm B class
  A
  I'm A class

分析:输出顺序:父类静态代码块>子类静态代码块>父类非静态代码块>父类构造器>子类非静态代码块>子类构造器;

区别:
静态代码块,在虚拟机加载类的时候就会加载执行,而且只执行一次;
非静态代码块,在创建对象的时候(即new一个对象的时候)执行,每次创建对象都会执行一次

分析:对象的初始化顺序:首先执行父类静态的内容,父类静态的内容执行完毕后,接着去执行子类的静态的内容,当子类的静态内容执行完毕之后,再去看父类有没有非静态代码块,如果有就执行父类的非静态代码块,父类的非静态代码块执行完毕,接着执行父类的构造方法;父类的构造方法执行完毕之后,它接着去看子类有没有非静态代码块,如果有就执行子类的非静态代码块。子类的非静态代码块执行完毕再去执行子类的构造方法。总之一句话,静态代码块内容先执行,接着执行父类非静态代码块和构造方法,然后执行子类非静态代码块和构造方法。 
而且子类的构造方法,不管这个构造方法带不带参数,默认的它都会先去寻找父类的不带参数的构造方法。如果父类没有不带参数的构造方法,那么子类必须用supper关键子来调用父类带参数的构造方法,否则编译不能通过。