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

代码示例:

情况一:没有继承父类时

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关键子来调用父类带参数的构造方法,否则编译不能通过