在 Java(TM) 编程语言中,变量的初始化依赖于其所在的上下文。请看下面代码:
int x;
Dog fido;
如果 x 和 fido 都是实例变量,他们就分别被自动初始化为 0 和 null 。null 是一个特殊的字面值,它能赋值给任何引用变量。
一般说来,如果实例变量在定义的时候没有被显式的初始化,那么 Java 将变量自动初始化为 "似零" 的值,具体值由变量的数据类型决定:
请看下面这段程序:
public class TestVariableInit {
public static void main(String[] args) {
TestClass tc = new TestClass();
System.out.println("tc.iStr = " + tc.iStr);
System.out.println("tc.i = " + tc.i);
}
}
class TestClass {
int i; // instance variable
String iStr; // instance variable
}
TestVariableInit 产生输出:
D:\>java TestVariableInit
tc.iStr = null
tc.i = 0
有些书上说,这种默认初始化在所有的情况下都适用,例如,一个局部变量(在方法体内声明的变量),但是局部变量如果没有显式初始化,根据所在 Java 环境不同,编译器会给出一个警告或者是报出一个错误。
实际情况是,以前有些Java环境对局部变量不提供初始化,也不给出警告或者错误,在这种情况下,值是垃圾,这可能造成非常难于诊断的运行错误。
在 Sun 微系统公司发布的 Java 开发包(JDK)中,编译器将在未初始化的局部变量的引用位置处报错。假如我们将 TestVariableInit 中的 main() 方法修改入下:
public static void main(String[] args) {
TestClass tc = new TestClass();
String lStr; // local variable
int i; // local variable
System.out.println("tc.iStr = " + tc.iStr);
System.out.println("tc.i = " + tc.i);
System.out.println("lStr = " + lStr);
System.out.println("i = " + i);
}
试图编译 TestVariableInit 会产生下面的输出:
D:\>javac TestVariableInit.java
TestVariableInit.java:8: Variable lStr may not
have been initialized.
System.out.println("lStr = " + lStr);
^
TestVariableInit.java:9: Variable i may not
have been initialized.
System.out.println("i = " + i);
^
2 errors
对于此类情况,Java 环境保证给出一个默认值,但是不允许你使用它,真是可笑!(如果一棵树掉到森林里,如果没有人听到落地声难道说它就没有发出声音吗?)
大部分的程序员都赞成为了可读性更好,实例变量应该初始化,并且局部变量必须初始化,这是为了使其他的程序员思路更清楚,也就是说,不管在哪儿,哪种 Java 环境中,源代码都可以被编译,即便是有一天谁都不知道代码的编写者是谁的时候。