29
public class Foo {
    public static void main(String[] args) {
        float f;
        System.out.println(f);
    }
}

print 语句会导致以下编译时错误,

局部变量 f 可能尚未初始化

如果 Java 中的原语已经有一个默认值 (float = 0.0f),为什么我需要定义一个?


编辑:

所以,这行得通

public class Foo {
    float f;
    public static void main(String[] args) {
        System.out.println(new Foo().f);
    }
}

谢谢大家!

4

5 回答 5

52

因为它是一个局部变量。这就是为什么没有分配给它的原因:

局部变量略有不同;编译器永远不会为未初始化的局部变量分配默认值。如果您无法在声明它的地方初始化局部变量,请确保在尝试使用它之前为其分配一个值。访问未初始化的局部变量将导致编译时错误。

编辑:为什么 Java 会引发此编译错误? 如果我们查看IdentifierExpression.java类文件,我们会发现这个块:

...
if (field.isLocal()) {
            LocalMember local = (LocalMember)field;
            if (local.scopeNumber < ctx.frameNumber && !local.isFinal()) {
                env.error(where, "invalid.uplevel", id);
            }
            if (!vset.testVar(local.number)) {
                env.error(where, "var.not.initialized", id);
                vset.addVar(local.number);
            }
            local.readcount++;
        }
...

如前所述 ( if (!vset.testVar(local.number)) {),JDK 检查 ( testVar) 是否分配了变量 (Vset的源代码,我们可以在其中找到testVar代码)。var.not.initialized如果不是,它会从属性文件中引发错误:

...
javac.err.var.not.initialized=\
    Variable {0} may not have been initialized.
...

来源

于 2012-06-22T23:43:59.727 回答
17

事实上,编译器并没有为你的 分配一个默认值float f,因为在这种情况下它是一个局部变量——而不是一个字段:

局部变量略有不同;编译器永远不会为未初始化的局部变量分配默认值。如果您无法在声明它的地方初始化局部变量,请确保在尝试使用它之前为其分配一个值。访问未初始化的局部变量将导致编译时错误。

于 2012-06-22T23:43:44.960 回答
8

类字段(final无论如何都不是)被初始化为默认值。局部变量不是。

声明字段时并不总是需要赋值。已声明但未初始化的字段将由编译器设置为合理的默认值。

所以一个(非final)字段f,如

class C {
  float f;
}

将被初始化为0f但局部变量f

void myMethod() {
  float f;
}

不会是。

语言对局部变量的处理与字段不同。局部变量的生命周期范围很广,因此在初始化之前的任何使用都可能是一个错误。字段没有,因此默认初始化通常很方便。

于 2012-06-22T23:43:33.110 回答
5

实际上局部变量存储在堆栈中。因此有可能获取局部变量存在的任何旧值。出于安全原因,这是一个很大的挑战..因此java说你必须在使用前初始化一个局部变量。

于 2013-07-04T07:35:44.517 回答
2

嗨,伙计们,解决方案很简单。存储在堆内存中的值由基于编译器的数据类型初始化,但局部变量存储在堆栈内存中,因此我们必须显式初始化它。

于 2015-02-19T07:22:22.293 回答