`
ZangXT
  • 浏览: 116491 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

一个静态初始化的问题

阅读更多

问题代码: 

public class Test1{

  1. public static int k=0;
  2. public static Test1 t1=new Test1("t1");
  3. public static Test1 t2=new Test1("t2");
  4. public static int i=print("i");
  5. public static int n=99;
  6. public int j=print("j");
  7. {
  8. print("构造块");
  9. }
  10. static{
  11. print("静态块");
  12. }
  13. public Test1(String str){
  14. System.out.println((++k)+":"+str+"i="+i+"n="+n);
  15. ++i;++n;
  16. }
  17. public static int print(Stringstr){
  18. System.out.println((++k)+":"+str+"i="+i+"n="+n);
  19. ++n;
  20. return ++i;
  21. }
  22. public static void main(String...strings){
  23. }
  24. }
  25. 输出结果:
  26. 1:ji=0n=0
  27. 2:构造块i=1n=1
  28. 3:t1i=2n=2
  29. 4:ji=3n=3
  30. 5:构造块i=4n=4
  31. 6:t2i=5n=5
  32. 7:ii=6n=6
  33. 8:静态块i=7n=99

问题:i和n到底是什么时候声明的,public static Test1 t1=new Test1("t1");这句会去执行构造块和构造方法,但是这时候public static int i=print("i");并没执行,i是哪里出来的?它是按顺序先声明所有变量然后才从头再挨个赋值吗?

原因是:类的生命周期分为如下几个阶段,Loading,Linking,Initialization.
Loading之后会对class文件进行Verification(验证).然后是Preparation阶段,注意在这个阶段所有的类变量(class变量),会被默认初始化,int,long,short,浮点等基本类型初始化为0,引用初始化为null,boolean被初始化为 false.也就是到这个阶段各个static变量已经有默认的值了;后面会进入Resolution阶段,然后才是初始化( Initialization),这个阶段会执行static初始化,把变量初始化为你指定的值.也就是在这个阶段k=0;Test1 t1=new Test1("t1");才开始执行的,这时候i=print("i");还没有执行到,i仍然取在 Preparation阶段得到的默认值0.

参考: 深入java虚拟机
http://www.artima.com/insidejvm/ed2/lifetype.html

明白了这个问题,静态初始化语句使用较多的时候一定要注意保证正确性,合理组织static语句的顺序.

分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics