Java类初始化中static代码块、构造代码块、构造函数的顺序

2021-11-20  本文已影响0人  我要离开浪浪山

1、顺序

父类静态块-->子类静态块--> main方法-->父类构造代码块-->父类构造函数-->子类构造代码块-->子类构造函数,同一级别代码块按顺序执行。

这几个名词体现在代码中,如下:

class Study {
    public Study(){
        System.out.println("构造函数");    
    }
    {
        System.out.println("构造代码块");    
    }
    static {
        System.out.println("静态代码块");        
    }
}

结果:
静态代码块--> 构造代码块--> 构造函数

2、静态代码块和代码块的区别在于:

静态代码块只执行一次,代码块每次创建该对象的时候都会执行一次。

3、普通情况

public class Demo {    
    static int a = 10;
    static {
         System.out.println("a----"+a);
         System.out.println("A的静态代码块");
    }
    public static void main(String[] args){
         System.out.println("main");
    }
}

输出结果:
a----10
A的静态代码块
main
public class Demo { 
    static int a = 10;
    static {
         System.out.println("a----"+a);
         System.out.println("A的静态代码块");
    }
    int b = 20;
    {
         System.out.println("b----"+b);
         System.out.println("A的构造代码块");
    }
    Demo(){
        System.out.println("构造函数");
    }
    public static void main(String[] args){
         System.out.println("main");
         new Demo();
    }
}

输出结果:
a----10
A的静态代码块
main
b----20
A的构造代码块
构造函数
public class Demo { 

    static {
        System.out.println("子类的静态代码块");
    }
    
    {
        System.out.println("子类的构造代码块");
    }
    Demo(){
        System.out.println("子类构造函数");
    }

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

结果:
子类的静态代码块
子类的构造代码块
子类构造函数
子类的构造代码块
子类构造函数

4、为什么只静态代码块只执行了一次,普通代码块执行了多次?

静态变量被所有的对象所共享,在内存中只有一个副本,它当且仅当在类初次加载时会被初始化。而非静态变量是对象所拥有的,在创建对象的时候被初始化,存在多个副本,各个对象拥有的副本互不影响。

5、继承情况

class Super {
    static {
        System.out.println("Super的静态代码块");
    }
    {
        System.out.println("Super的构造代码块");
    }
    Super(){
        System.out.println("Super构造函数");
    }

}

public class Demo extends Super{
    static {
        System.out.println("子类的静态代码块1");
    }

    static {
        System.out.println("子类的静态代码块2");
    }
    
    {
        System.out.println("子类的构造代码块");
    }
    Demo(){
        System.out.println("子类构造函数");
    }

    public static void main(String[] args){
        System.out.println("main");
        new Demo();
    }

}

输出:
Super的静态代码块
子类的静态代码块1
子类的静态代码块2
main
Super的构造代码块
Super构造函数
子类的构造代码块
子类构造函数

6、为什么会调用父类的构造函数呢?命名没有调用过Super方法。

子类构造函数会默认调用一下父类的无参构造函数,可以给父类构造函数加个参数试一下。基础知识不能忘啊。

7、加载过程:

在执行开始,先要寻找到main方法,因为main方法是程序的入口,但是在执行main方法之前,必须先加载Demo类,而在加载Demo类的时候发现Demo类继承自Super类,因此会转去先加载Super类,在加载Super类的时候,发现有static块,便执行了static块。在Super类加载完成之后,便继续加载Demo类,然后发现Demo类中也有static块,便执行static块。在加载完所需的类之后,便开始执行main方法。在main方法中执行new Demo()的时候会先调用父类的构造器,然后再调用自身的构造器。因此,便出现了上面的输出结果。
同时存在两个代码块的时候,按代码顺序执行。

8、示例 1

public class Demo {
    private static Demo test = new Demo();
    //静态变量sta1    未赋予初始值
    public static int sta1;
    //静态变量sta1    赋予初始值20
    public static int sta2 = 20;
    //构造方法中对于静态变量赋值
    private Demo() {
        sta1 ++ ;
        sta2 ++ ;
    }
    public static void main(String[] args) {
        System.out.println(Demo.sta1);
        System.out.println(Demo.sta2);
    }
}

print:
1
20
解释:都是static就顺序执行,执行完后,让sta1和2自增,完后sta2又赋值为20。

9、示例 2

构造代码块和对象绑定,简单讲,不new对象不走构造代码块。

public class Animal {
    private static int k;
    static{
        System.out.println("父类的静态方法");
    }
    {
        System.out.println("执行父类的构造代码块");
    }
    public Animal(){
        System.out.println("执行父类的构造方法");
    }
    public static void main(String[] args) {
        System.out.println(Animal.k);
    }
}
print:
父类的静态方法
0

这个链接里面也有几道烧脑的题目,反正就按顺序来就好了,但是写代码的时候最好还是以大部分人能看明白为主,要不然光这个点很多人就得进坑,而且时间久了自己也会记不牢。
https://www.cnblogs.com/chihirotan/p/6043442.html


参考:https://www.jianshu.com/p/0c4223e0702b

上一篇 下一篇

猜你喜欢

热点阅读