记一次BUG:static关键字

序言:近期朋友提到了一个有关于static关键字一个坑,今天把它总结一下,希望能给读者解惑。在阅读之前请先对static不同使用场景的概念有所了解。

Code

public class Test {
    private static Test t1 = new Test();

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

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

    public static void main(String[] args) {
        Test t2 = new Test();
    }
}

问题:以上代码的打印结果是什么?

blockA
blockB
blockA

分析

根据我个人的理解一个代码块被加载的顺序是:静态代码块‘优先于’代码块,而以上的打印却反之。
通过查看编译后的字节码文件如下:

public class Test {
    private static Test t1 = new Test();

    public Test() {
        System.out.println("blockA");
    }

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

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

由此可见,普通的代码块直接被编译到了无参构造器中。
现在的分析是:在实例化一个Test静态对象时,先走了构造函数,再走静态代码块,所以打印blockA、blockB。而在函数中又实例化一个Test对象(这时类已经被加载一次,也就是说static修饰的成员变量,静态代码块等等之后被加载一次),所以在打印一个blockA。
以上的解释似乎能走通,但是存在疑问:

  • 根据类加载的执行顺序,静态修饰的都要先走,非静态后走,现在的解释相反!现在又猜想构造器是静态的?
    • 如果是静态的就违背了Java语言规范
    • 静态方法里面是不可以使用this,而构造函数可以使用this
    • 静态方法是不可以重写的,构造方法确实不可以重写

带着疑问我翻阅了Java编程思想和bing了许多资料。在Java编程思想中的第96页,根据书的作者所说Java的构造函数可以看成静态的;当然网上也有很多反例,比如:通过jvm底层去研究。
看了这么多,个人总结为:构造函数不是静态的也不是非静态的,就像正负数与0一样。

此文仅为个人总结,如有不对之处,请在留言指出,谢谢!

猜你喜欢

转载自blog.csdn.net/qq_34382367/article/details/86634464