当前位置: 首页 > 面试题库 >

为什么局部类会接受静态最终变量?

谢俊悟
2023-03-14
问题内容

我已经在Google上进行了广泛的搜索,但均无济于事。我似乎无法将这个概念笼罩在脑海中。为什么本地班级会接受静态最终字段?如下面的以下示例

public void sayGoodbyeInEnglish() {

        class EnglishGoodbye {
            public static final String farewell = "Bye bye";
            public void sayGoodbye() {
                System.out.println(farewell);
            }
        }
        System.out.println(EnglishGoodbye.farewell);
        EnglishGoodbye myEnglishGoodbye = new EnglishGoodbye();
        myEnglishGoodbye.sayGoodbye();
    }

在EnglishGoodbye课堂上,允许告别可变吗?为什么?我很困惑。为什么允许但没有静态变量?我知道为什么它不能访问封闭范围的成员,除非它们是一个编译器时间常数,因为当函数结束时这些变量不再存在,但类可能不存在。对?我只是对此感到困惑。

谢谢!


问题答案:

一般而言,不是。

但是farewell是一种特殊的静态最终形式:其值是一个常量,如JLS
15.28
所定义。这意味着它没有在该位置被初始化,这是根据JLS
8.1.3
在非静态类(包括本地类)中实际上不允许的。

JLS在8.1.3中明确指出了这一点(并以粗体显示)(请注意“除非”部分):

如果内部类声明一个显式或隐式静态成员,则将导致编译时错误,除非该成员是常量变量(第4.12.4节)。

如果您更改该行以删除final修饰符或使表达式非恒定(例如new String("Bye bye")),那么您将得到预期的编译错误:

Test.java:5: error: Illegal static declaration in inner class EnglishGoodbye
            public static final String farewell = new String("Bye bye");
                                       ^
  modifier 'static' is only allowed in constant variable declarations
1 error

多一点:

允许这样做的原因是常量变量由编译器专门处理。特别是,可以内联它们-生成的字节码根本没有该farewell字段!如果您对类(javap -c YourClassName)进行反编译,则会看到以下内容:

public void sayGoodbyeInEnglish();
  Code:
     0: getstatic     #2                  // Field java/lang/System.out:Ljava/io/PrintStream;
     3: ldc           #3                  // String Bye bye
     5: invokevirtual #4                  // Method java/io/PrintStream.println:(Ljava/lang/String;)V
     ...

上面对应于这一行:

System.out.println(EnglishGoodbye.farewell);

这有点令人生畏,但请注意“ 3:”行。该程序没有加载该字段的值farewell,而是加载了常量3(它在注释中指出是字符串“ Bye
bye”)(您可以在Wikipedia上看到字节码的列表)。

因为farewell是一个常量变量(而不是“真正的”静态成员),因此可以内联到代码中,所以在哪里定义它都无所谓-
变量的生命周期本质上是整个JVM的生命周期,而不是任何一个类或实例,因此可以在任何地方声明。



 类似资料:
  • 问题内容: 在C / C ++中,我们使用静态局部变量来维护方法的状态。但是,为什么Java不支持它呢? 是的,我可以为此使用一个静态字段。但是创建一个仅维护一个方法状态的字段有点奇怪吗? 问题答案: 您已经找到了唯一的解决方案。 Java放弃了C ++的许多复杂性,这就是其中之一。 作用于函数的静态变量并发地对您造成麻烦(例如,正是由于这个原因,strtok是与pthread一起使用的著名的讨厌

  • 本文向大家介绍接口变量在Java中默认为静态且为最终变量,为什么?,包括了接口变量在Java中默认为静态且为最终变量,为什么?的使用技巧和注意事项,需要的朋友参考一下 接口定义行为的协议,而不是行为的协议。实现接口的类遵守该接口定义的协议。 接口变量是静态的,因为无法单独实例化Java接口。必须在没有实例的静态上下文中分配变量的值。 final修饰符确保分配给接口变量的值是无法重新分配的真实常量。

  • 问题内容: 我什至不知道此代码即使能正常工作也不会做任何事情,但是我不知道该怎么做才能摆脱“从内部类引用的局部变量必须是最终的或实际上是最终的”错误消息,该错误消息在以“ fireballRight [i]”开头的三行中显示。 任何指导将不胜感激,谷歌似乎没有帮助我这一点。 问题答案: 您尚未显示所有代码,但我怀疑添加了以下内容: 在循环内使用而不是将其用作数组的索引应该可以修复错误。 另外,如@

  • 问题内容: 请参考下面的代码。运行代码时,我可以更改最终非静态变量的值。但是,如果我尝试更改最终静态变量的值,则会抛出异常。 我的问题是,为什么在非静态最终变量也不会抛出异常,反之亦然。为什么会有所不同? 问题答案: 该解决方案并非没有缺点,它可能无法在所有情况下都有效: 如果在字段声明中将字段初始化为编译时常量,则对该字段的更改可能不可见,因为该最终字段的使用会在编译时用编译时常量替换。 另一个

  • 问题内容: 给定下面的Java代码,您可以在Ruby类中最接近地表示这两个变量吗?而且,在Ruby中是否可以像Java中那样区分和变量? 问题答案: Ruby中确实没有等效的构造。 但是,您似乎犯了一个经典的移植错误:您有一种使用语言A 的 解决方案 ,然后尝试将其翻译为语言B,这时您真正应该做的就是找出 问题 ,然后找出解决方法使用语言B。 我不能完全确定您要从那个小的代码段中解决什么问题,但是

  • 问题内容: 为什么默认情况下接口变量是静态变量和最终变量? 问题答案: 从Philip Shaw的Java接口设计常见问题解答中: 接口变量是静态的,因为无法单独实例化Java接口。必须在没有实例的静态上下文中分配变量的值。final修饰符确保分配给接口变量的值是一个真正的常量,不能由程序代码重新分配。