日期:2014-05-20  浏览次数:20823 次

JAVA中static成员变量赋值的神奇问题
  1
  2 class data{
  3         private static final data mInstance = new data();
  4         private static int mx = 0;
  5         private data()
  6         {
  7                 System.out.println("mx=" + mx);
  8                 mx = getVersion();
  9         }
 10         public static data getInstance()
 11         {
 12                 return mInstance;
 13         }
 14         public int getVersion()
 15         {
 16                 int v = 0;
 17                 if (v == 0)
 18                 {
 19                         v = 7;
 20                 }
 21                 return v;
 22         }
 23         public int getmx()
 24         {
 25                 return mx;
 26         }
 27 }
 28
 29 class Test
 30 {
 31         public static void getData(StringBuffer sb)
 32         {
 33                 sb.append("hello world ");
 34         }
 35         public static void main(String args[])
 36         {
 37                   System.out.println(data.getInstance().getmx());
 38         }
 39 }
~
~
打印结果:
mx=0
0
这样的结果怎么得到了,求教

          
------解决方案--------------------
原理就是在执行此static initializer时:
private static final data mInstance = new data();

jvm已经知道初始化体中所涉及变量的类型了(感觉类型初始化早就完成了),这里就是mx被当做int型了,且被赋予了int型的默认值0

最后还会执行
private static int mx = 0;

但看来仅仅相当于mx=0
赋值而已
------解决方案--------------------
其实这个问题很简单,牵扯到类加载顺序以及类中代码执行的顺序问题。
类的初始化顺序如下:
父类静态变量
父类静态块
子类静态变量
子类静态块
父类变量
父类普通块
父类构造函数(子类实例化时先要调用父类构造函数)
子类变量
子类普通块
子类构造函数

如果有多个初始化块,则按照代码先后顺序执行。
因此此问题解释如下:
1.date中的静态变量data会在类加载的时候首先初始化,因此初始化的时候调用date构造函数,将会将m赋值为7,
2.然后执行private static int mx = 0进行初始化m操作,因此此时m又成了0.
楼主可以将private static int mx = 0这句放在private static final data mInstance = new data();前面执行,结果就会不一样了。

另外楼主可以在private static final data mInstance = new data();
后面加上
static{
System.out.println("static");
}
然后执行看看初始化的先后顺序如何

------解决方案--------------------
就是一个执行顺序的问题
3 private static final data mInstance = new data(); //这里会调用getVersion使得mx为7
4 private static int mx = 0;//但是这里又重新给mx赋值为0了

------解决方案--------------------
引用:
引用:

原理就是在执行此static initializer时:
private static final data mInstance = new data();

jvm已经知道初始化体中所涉及变量的类型了(感觉类型初始化早就完成了),这里就是mx被当做int型了,且被赋予了int型的默认值0

最后还会执行
private sta……

我也不太清楚 只是通过这个案例才知道了一些
如果感兴趣可以去看深入jvm这本书
如果改成那样,mx最后就等于7了

从这里似乎可以得到一个结论:初始化中的类型初始化部分是最先完成的,都给一个对应类型的默认初始值,然后再去按顺序执行赋值操作,没有的就不执行了