今天看到了一段代码,很有意思。
public class Test {
static{
i=0;
System.out.println(i); // 编译错误"Cannot reference a field before it is defined"
}
public static int i=1;
}
上面一段代码,输出语句报编译错误,报错信息翻译一下 “无法在定义字段之前引用该字段”,乍一看,这错报的确实没啥问题啊,i 还没定义,咋能输出它呢?
但是我把这段输出语句注释掉,发现程序又正常运行了,也就是这样
public class Test {
static{
i=0;
}
public static int i=1;
}
静态变量 i 明明还没有声明,为什么可以在静态代码块中做 i = 0 这个赋值操作呢?
我通过面向CSDN编程,解开了这一疑惑。static修饰的变量的声明和赋值其实是分开的。比如 public static int i=1; 在编译期JVM就已经把 public static int i 这一声明操作抽了出来并且执行了。也就是说在加载类之前,静态变量 i 就已经被声明了,内存中已经开辟出了属于它的空间。上面一段代码,其实可以抽象为这个样子:
编译期:public static int; //编译期就已经声明了静态变量i
public class Test {
static{ //运行期就只剩赋值操作
i=0;
i=1;
}
}
这也就是为什么 i=0; 可以先于 public static int i=1; 执行而不报错。
但是问题又来了。众所周知,想要使用一个变量,要声明它并且对它赋值。根据上面的结论,我们这段代码中静态变量 i 在运行期就已经被声明了,i=0; 这行代码又执行了赋值操作,那么 i 不就可以用了吗?为什么输出 i 还是会报编译错误?
我的个人理解是,按照理论来说,上面这种情况中,i 确实可用。但是先使用变量,后声明变量,这跟java的语法规则是背道而驰的!虽然理论上可行,但是它终究还是违背了java的语法规则,所以编译报错。
虽然静态变量的声明赋值很特殊,可以 “先赋值,后声明”,但是我们书写静态变量和静态代码块时最好仍然遵守java语法规则,声明在前,赋值在后,避免不必要的错误。按规矩办事,你好我好大家好!
最后贴出一位大牛的帖子,从根源上解释了这个问题。希望我有一天能达到这位大牛的境界哈哈:
https://blog.csdn.net/darxin/article/details/5293427