相关疑难解决方法(0)

从lambda表达式引用final字段

最近我发现了匿名类和lambda表达式之间的细微差别:

public class FinalTest {
    final Runnable x = new Runnable() {
        @Override
        public void run() {
            System.out.println(x.hashCode());
        }
    };

    final Runnable y = () -> System.out.println(y.hashCode()); 
}
Run Code Online (Sandbox Code Playgroud)

通常lambdas等同于匿名类.甚至我的Eclipse IDE都有重构转换x为lambda(它变得完全像y)并转换y为匿名类(它变得完全一样x).然而lambda给了我一个编译错误,而匿名类可以完美编译.错误消息如下所示:

>javac FinalTest.java
FinalTest.java:9: error: self-reference in initializer
    final Runnable y = () -> System.out.println(y.hashCode());
                                                ^
1 error
Run Code Online (Sandbox Code Playgroud)

所以问题是:为什么会有这样的差异?

java javac java-8

11
推荐指数
1
解决办法
423
查看次数

Java:为什么在定义字段之前引用字段时没有警告?

在定义或初始化静态字段之前,不能引用静态字段:

static Integer j = i; /* compile error */
static final Integer i = 5;
Run Code Online (Sandbox Code Playgroud)

但是,当从实例初始化块(在匿名内部类中)引用它时,甚至不会生成警告.

见例子:

class StaticInitialization {

    static final Object o = new Object() {{
        j = i;
    }};

    static Integer j, k;
    static final Integer i = 5;

    static final Object o2 = new Object() {{
        k = i;
    }};
}
Run Code Online (Sandbox Code Playgroud)

结果是:j == null,k == 5显然我们已经做了引用,命令很重要,没有警告或编译错误.

这段代码合法吗?

java static initialization inner-classes

10
推荐指数
1
解决办法
285
查看次数

标签 统计

java ×2

initialization ×1

inner-classes ×1

java-8 ×1

javac ×1

static ×1