Java 8 lambdas, Function.identity()或t->t

我有一个关于Function.identity()方法使用的问题。

想象下面的代码:

Arrays.asList("a", "b", "c")
.stream()
.map(Function.identity()) // <- This,
.map(str -> str)          // <- is the same as this.
.collect(Collectors.toMap(
Function.identity(), // <-- And this,
str -> str));        // <-- is the same as this.

是否有任何理由你应该使用Function.identity()而不是str->str(反之亦然)。我认为第二种选择更具有可读性(当然这是一种品味问题)。但是,有什么“真正”的理由让一个人更受青睐吗?

197480 次浏览

在你的例子中,str -> strFunction.identity()之间没有太大的区别,因为在内部它只是t->t

但有时我们不能使用Function.identity,因为我们不能使用Function。看看这里:

List<Integer> list = new ArrayList<>();
list.add(1);
list.add(2);

这样编译就可以了

int[] arrayOK = list.stream().mapToInt(i -> i).toArray();

但是如果你尝试编译

int[] arrayProblem = list.stream().mapToInt(Function.identity()).toArray();

你会得到编译错误,因为mapToInt期望ToIntFunction,这与Function无关。而且ToIntFunction没有identity()方法。

JDK源:

static <T> Function<T, T> identity() {
return t -> t;
}

所以,只要语法正确就可以。

在当前的JRE实现中,Function.identity()总是返回相同的实例,而每次出现identifier -> identifier不仅会创建自己的实例,而且甚至有一个不同的实现类。更多细节,参见在这里

原因是编译器生成一个合成方法,保存该lambda表达式的普通主体(在x->x的情况下,相当于return identifier;),并告诉运行时创建调用该方法的函数接口的实现。因此,运行时只能看到不同的目标方法,而当前实现不会分析这些方法,以确定某些方法是否等效。

因此,使用Function.identity()而不是x -> x可能会节省一些内存,但如果你真的认为x -> xFunction.identity()更可读,那就不应该影响你的决定。

你也可以考虑在启用调试信息的情况下编译时,合成方法将有一个行调试属性,指向持有lambda表达式的源代码行,因此你有机会在调试时找到特定Function实例的源。相反,当在调试操作期间遇到Function.identity()返回的实例时,您将不知道是谁调用了该方法并将该实例传递给该操作。