序言:近期朋友提到了一个有关于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一样。此文仅为个人总结,如有不对之处,请在留言指出,谢谢!