返回介绍

鲜为人知的 Java8 特性:泛化目标类型推断

发布于 2025-02-26 22:54:42 字数 2816 浏览 0 评论 0 收藏 0

在浏览 Java8 的特性列表 的时候, 目标类型推断 这个特别有趣的、鲜为人知的特性一下子吸引了我。Java 语言的设计者通过它让我们减轻了一些使用泛型时(Java5-7)的痛苦。让我们来看看过去泛型使用的示例:

class List<E> {
  static <Z> List<Z> nil() {..}
  static <Z> List<Z> cons(Z head, List<Z> tail) {..}
  E head() {..}
}

在上述例子,在 JEP:101 中声称可以用下面的方法更好地表示:

// 建议写法:
List.cons(42, List.nil());
String s = List.nil().head();

// 不推荐的写法:
List.cons(42, List.<Integer>nil());
String s = List.<String>nil().head();

作为一个 熟练的 API 设计师 ,在 Java 路线图中看到示例中的进步着实令人激动。这些令人兴奋的变化究竟包含了什么?让我来更加详细地说明:

// 通过赋值语句推断泛型的类型
List<String> l = List.nil();

// 更好的办法是让编译器从函数的参数类型中直接推断
List.cons(42, List.nil());

// 或者从“链式调用”中推断
String s = List.nil().head();

因此在上面的链式方法调用中,会延迟到整个赋值表达式完成时才进行类型推断。通过赋值语句左边,编译器会为 head() 调用推断;为 String 。然后,再次推断 nil() 调用的为 String 。 在我看来这真的很神奇。 对 nil() 方法的 AST 计算会延迟到“关联”子节点计算时才最后完成。这是一个很棒的主意,不是吗?

是的,确实很棒!

你可能也会这么认为。因为一组流畅的 API,像 jooq 或 Stream API 在设计时会考虑到这种调用的流畅性,在链式调用的最后才进行类型推断。为此,我下载了最新的 JDK 8 评估版本测试下面的程序:

public class InferenceTest {
    public static void main(String[] args) {
        List<String> ls = List.nil();
        List.cons(42, List.nil());
        String s = List.nil().head();
    }
}

以下是得到的编译结果:

C:\Users\Lukas\java8>javac InferenceTest.java
InferenceTest.java:5: error: incompatible types: 
    Object cannot be converted to String
        String s = List.nil().head();
                                  ^
1 error

从结果中可以看到,基于该方法参数的类型推断已经实现了(因此编译通过了),但是链式方法调用中的类型推断还没有实现。我在网上搜索到了一个解释,从 Stack OverFlow 问题 链接到 lambda-dev 开发者邮件列表 中。

看来,Java 类型系统已经变得相当复杂。由于太过复杂,要实现这种疯狂的类型推断变得不太现实。但是,每天编写 Java 8 代码的时候,即使略有改善也有重大的价值。

最后,希望在 Java 9 中会有 val 和 var 这样的关键字,与其他语言一样。

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据
    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文