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

javac为什么不能为用作参数的函数推断泛型类型参数?

曹和正
2023-03-14
问题内容

在下面的示例中,为什么编译器能够为Foo.create()in中的第一次调用推断出通用参数,而在第二次调用中Foo.test()却无法推断出通用参数?我正在使用Java
6。

public class Nonsense {
    public static class Bar {
        private static void func(Foo<String> arg) { }
    }

    public static class Foo<T> {

        public static <T> Foo<T> create() {
            return new Foo<T>();
        }

        private static void test() {
            Foo<String> foo2 = Foo.create(); // compiles
            Bar.func(Foo.create());          // won't compile
            Bar.func(Foo.<String>create());  // fixes the prev line
        }
    }
}

(编译错误为 Nonsense.Bar类型的func(Nonsense.Foo)方法不适用于参数(Nonsense.Foo) )。

注意:我了解编译器错误可以通过test()中的第三行来解决-我很好奇是否存在阻止编译器推断类型的特定限制。这 似乎 对我有足够的上下文在这里。


问题答案:

从Java
7开始,必须考虑方法重载解析,然后才能考虑您正在调用的方法中的任何目标类型信息,以尝试推断T的声明中的类型变量func。这似乎很愚蠢,因为我们所有人都可以看到在这种情况下只有一个名为的方法func,但是它是JLS强制执行的,并且是javacJava
7 的行为。

编译过程如下:首先,编译器看到它正在编译对名为Func的Bar类的静态方法的调用。为了执行重载解析,它必须找出调用该方法参数。尽管这是一个微不足道的案例,但它仍然必须这样做,直到这样做为止,它没有有关可用于帮助它的方法的形式参数的任何信息。实际参数由一个参数组成,对该参数的调用Foo.create()被声明为returning
Foo<T>。再次,没有来自目标方法的标准,它只能推断出该返回类型是擦除的Foo<T>Foo<Object>,它也如此。

然后,方法重载解析将失败,因为的重载都不func与的实际参数兼容Foo<Object>,并且会因此产生错误。

这当然是非常不幸的,因为我们所有人都可以看到,如果信息可以简单地在另一个方向上流动(从方法调用的目标返回调用位置),则可以轻松推断出类型,并且不会出错。实际上,Java
8中的编译器可以做到,而且可以做到。另一个答案是,这种丰富的类型推断对Java 8中添加的lambda和对利用lambda的Java API扩展非常有用。

您可以从前面的链接下载带有JSR 335 lambda的Java
8的预发布版本。它在没有任何警告或错误的情况下编译问题中的代码。



 类似资料:
  • 我想使用泛型类作为另一个泛型类的类型参数。 起初,我对类的定义是这样的: 然后我的需求发生了变化,我不得不为我的R类型使用包装器/持有者类 到目前为止,我的尝试:(给出编译时错误:

  • 我需要 Kotlin 中的一个集合来仅包含实现给定接口的元素。 例如:包含动物集合的地图: 通过阅读文档、博客和SO问题,我编写了使用Generics in关键字的代码: 现在我想在Test类中添加一个读取“data”内容的方法,例如将其打印到控制台: 如果我这样做,我会遇到编译错误: 我的解决方案是强制我的动物进入一个ArrayList 但是我不确定这是编写这种代码的最好方式。有没有更好的方式告

  • 问题内容: ./chains.go:26:10:不能在作业中使用UpperCaseHandler(typefunc(asl.MessageDelivery))作为asl.MessageHandler类型./chains.go:37:86:无法使用RepeatHandler(类型func(asl.MessageDelivery))与Repeater.ConsumeFunc的参数中的asl.Messa

  • 请按顺序阅读代码中的注释,问题详细信息在那里<为什么会出现这种差异 如果可能,请引用JLS。 我最初在IntelliJ IDEA中发现了这一点,但我猜编译器与javac兼容,因为当我用以下代码编译上述代码时,它给出了相同的错误/警告。

  • 定义参数化类时,只能使用固定数量的参数。 但是,如果你想创建一个包含多个值的地图。必须使用映射

  • 假设我们有以下代码: 一个相当简单的片段——我们获取一个地图,并返回另一个代表相同地图但根据其值排序的地图。 让我们稍微改变一下功能——与其基于参数返回一些内容,不如介绍一个对象: 请注意,我们唯一改变的是去掉参数,取而代之的是使用字段。有人可能想知道为什么要引入泛型和——毕竟它们总是分别是和。但是这难道不意味着Java可以很容易地推断出这些参数吗?为什么它会失败,错误为: 我看到了一些关于类似错