java - 为什么这个与 lambda 相关的类型推断会失败?

标签 java lambda java-8 type-inference

我想了解为什么以下内容无法处理“引用不明确”,而它似乎不应该如此。值得向 javac 团队报告吗?

import java.util.function.Function;
import java.util.function.ToLongFunction;
import org.junit.Test;

public class LambdaTest {

    @FunctionalInterface
    public static interface CheckedFunction<U,R>
    {
        R apply(U value) throws Exception;
    }

    @FunctionalInterface
    public static interface CheckedToLongFunction<T>
    {
        long apply(T i) throws Exception;
    }

    public static <T,R> Function<T,R> unchecked(CheckedFunction<T,R> func)
    {
        return value -> {
            try
            {
                return func.apply (value);
            }
            catch(RuntimeException e)
            {
                throw e;
            }
            catch(Exception e)
            {
                throw new RuntimeException(e);
            }
        };
    }

    public static <T> ToLongFunction<T> unchecked(CheckedToLongFunction<T> func)
    {
        return value -> {
            try
            {
                return func.apply (value);
            }
            catch(RuntimeException e)
            {
                throw e;
            }
            catch(Exception e)
            {
                throw new RuntimeException(e);
            }
        };
    }

      public void 
    bar( Function<Object,Object> fn ) { 

            System.out.println("Function"); 
        }

      @Test public void
    test() {

            bar (a -> a); //OK
            bar (unchecked (a -> a)); //Should be OK, but receive "reference to unchecked is ambiguous"
            bar (unchecked ((Object a) -> a)); //OK
        }

}

正如您所看到的,这个示例很有趣,因为 CheckedFunctionCheckedToLongFunction 都采用引用类型,因此显式指定amba的输入类型似乎没有添加任何有用的内容。然而,它可以让代码编译。

unckeched(CheckedFunction) 的调用应该没有歧义,因为 unchecked(CheckedToLongFunction) 生成一个返回原语的 lambda,而 bar 需要一个 Object.

我特别寻求 JLS 的解决方案。如果 JLS 没有指定这一点,我可以尝试将其归档为 javac bug。

最佳答案

你有两种方法

public static <T,R> Function<T,R> unchecked(CheckedFunction<T,R> func)
public static <T> ToLongFunction<T> unchecked(CheckedToLongFunction<T> func)

每个都有一个功能接口(interface)类型的参数。

@FunctionalInterface
public static interface CheckedFunction<U, R> {
    R apply(U value) throws Exception;
}

@FunctionalInterface
public static interface CheckedToLongFunction<T> {
    long apply(T i) throws Exception;
}

implicitly typed lambda expression如果有适当的上下文,a -> a 可以应用于其中任何一个。但是,在进行重载解析时似乎没有检查该上下文。

所以虽然

中的 a -> a
bar(unchecked(a -> a)); // Should be OK, but receive

无法转换为 CheckedToLongFunction 实例(因为无法从该调用上下文推断出 Long 的类型参数),lambda 仍然适用于函数类型。

编译器根本无法确定是否应将 a -> a 转换为 CheckedToLongFunction 还是 CheckedFunction。由于两者都适用,因此调用是不明确的。不会检查调用上下文(和返回类型)来解决歧义。

但是,在这种情况下,

bar(unchecked((Object a) -> a)); // OK

我们有一个显式类型 lambda 表达式。 lambda 主体解析为 Object 类型的返回值,这不符合 CheckedToLongFunction#apply(..) 方法的要求,即。 Object 类型的表达式无法隐式转换为 long 类型的表达式。

因此,由于还有另一种适用的方法,因此选择了该方法。

关于java - 为什么这个与 lambda 相关的类型推断会失败?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26203943/

相关文章:

java - 通过 XML 使用 SpringSocialConfigurer 添加 SocialAuthenticationFilter,以便通过社交提供商登录

java - 如何建立一个模型来区分关于 Apple (Inc.) 的推文和关于苹果 (fruit) 的推文?

java - JAVA 中带有 ResultSet 参数的 SELECT 查询

java - 在外部类的构造函数中实例化内部类是否危险? ( java )

lambda - 如何在 lisp 中本地定义一个函数?

java - 从数组中查找第二个最小元素

java - 获取目录中的文件名

c# - 为什么泛型 class<T> 中的 List<T>.RemoveAll 需要 Predicate 的变体形式?

java - Lambda 泛型类型界限

lambda - 对流的 groupBy 结果应用操作