下面的代码在 Java 7中编译并运行正常,但无法在Java 1.8.0中编译u25:
public class GenericTest {

    public static class Genericclass<T> {
        T value;

        public Genericclass(T value) {
            this.value = value;
        }
    }

    public static class SecondGenericclass<T> {
        T value;

        public SecondGenericclass(T value) {
            this.value = value;
        }
    }


    public static<T >void verifyThat(SecondGenericclass<T> actual,Genericclass<T> matcher) {
    }

    public static<T >void verifyThat(T actual,Genericclass<T> matcher) {
    }

    @Test
    public void testName() throws Exception {
        verifyThat(new SecondGenericclass<>(""),new Genericclass<>(""));
    }

}

Java 8中的错误消息如下所示:

Error:(33,9) java: reference to verifyThat is ambiguous
  both method <T>verifyThat(com.sabre.ssse.core.dsl.GenericTest.SecondGenericclass<T>,com.sabre.ssse.core.dsl.GenericTest.Genericclass<T>) in com.sabre.ssse.core.dsl.GenericTest and method <T>verifyThat(T,com.sabre.ssse.core.dsl.GenericTest.Genericclass<T>) in com.sabre.ssse.core.dsl.GenericTest match

我已经查看了以下所有更改:
https://docs.oracle.com/javase/specs/jls/se8/html/jls-15.html#jls-15.12.2
https://docs.oracle.com/javase/specs/jls/se7/html/jls-15.html#jls-15.12.2

但是我没有注意到这个行为的确切原因.

编辑:

只是为了回答一些意见,很清楚,Java 7和8中的编译器将能够处理这种调用(签名类似于编译时类型擦除之后留下的签名:

public static void verifyThat(SecondGenericclass actual,Genericclass matcher) {
}

public static void verifyThat(Object actual,Genericclass matcher) {
}

@Test
public void testName() throws Exception {
    verifyThat(new SecondGenericclass<>(""),new Genericclass<>(""));
}

为通用方法生成的字节码和已擦除的字节码是一样的,如下所示:

public static verifyThat(Lcom/sabre/ssse/core/dsl/GenericTest$SecondGenericclass;Lcom/sabre/ssse/core/dsl/GenericTest$Genericclass;)V
public static verifyThat(Ljava/lang/Object;Lcom/sabre/ssse/core/dsl/GenericTest$Genericclass;)V

EDIT2:

javac 1.8.0_40下的编译失败,出现相同的错误

解决方法

JLS,chapter §15.12.2.5 Choosing the Most Specific Method是一个难读,但包含一个有趣的总结:

The informal intuition is that one method is more specific than another if any invocation handled by the first method Could be passed on to the other one without a compile-time type error.

我们可以轻易地为您的案例反驳,具体如下:

GenericTest.<String>verifyThat( // invokes the first method
    new SecondGenericclass<>(""),new Genericclass<>(""));
GenericTest.<SecondGenericclass<String>>verifyThat( // invokes the second
    new SecondGenericclass<>(""),new Genericclass<>(null));

所以这里没有最具体的方法,但是,如示例所示,可以使用使其他方法不适用的参数来调用任一方法.

在Java 7中,由于编译器尝试(参见编译器)有限的寻找类型参数以使更多方法适用(也称为有限类型推断),因此更容易使方法不适用.新的SecondGenericclass(“”)表达式具有SecondGenericclass< String>从它的论证推断“”就是这样.因此,对于调用verifyThat(new SecondGenericclass(“”),新的Genericclass(“”)),参数的类型为SecondGenericclass< String>和Genericclass< String>这使得方法< T> void verifyThat(T,Genericclass 一个模糊调用的示例,其在java="" 7(甚至java="" 6)下展现了歧义):verifythat(null,null);使用javac时会引发编译器错误.="" 但是java="" 8有 Invocation Applicability Inference(我们与JLS 7有一个全新的篇章…),它允许编译器选择引用方法候选的类型参数(通过嵌套调用工作).您可以为您的特殊情况找到这样的类型参数,甚至可以找到适合两者的类型参数,>)不适用.>

GenericTest.<Object>verifyThat(new SecondGenericclass<>(""),new Genericclass<>(""));

毫无疑问(在Java 8中),即使Eclipse也同意这一点.相反,调用

verifyThat(new SecondGenericclass<>(""),new Genericclass<String>(""));

具体足以使第二种方法不适用并调用第一种方法,这给我们提供了一个关于Java 7中发生的情况的提示,其中新的Genericclass(“”)的类型被固定为Genericclass< String>就像新的Genericclass< String>(“”).

底线是,它不是从Java 7到Java 8(显着地)改变的最具体的方法的选择,而是由于改进的类型推断的适用性.一旦这两种方法都适用,调用是不明确的,因为这两种方法都不比其他方法更具体.

Java 8模糊方法引用通用类的更多相关文章

  1. ios – 如何使用Objective C类中的多个参数调用Swift函数?

    本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容,请发送邮件至dio@foxmail.com举报,一经查实,本站将立刻删除。

  2. ios – 在XCTestCase子类中使用泛型有效吗?

    我有一个XCTestCase子类,看起来像这样.为了简洁起见,我已经删除了setup()和tearDown方法:它的子类看起来像这样:在理论上,这应该按预期工作–编译器不会抱怨任何事情.但是只是当我运行测试用例时,setup()方法甚至没有被调用.但是,它表明当testName()方法应该失败时,测试已经过去了.使用泛型是一个问题吗?我可以很容易地想到很多非通用的方法,但是我很想写这样的测试用例.这是XCTest在Objective-C和Swift之间的互操作性?Ergo您的通用XCTestCase子类不

  3. ios – 使用反射访问模型类的静态变量

    我有一个包含一些静态变量和属性的模型类.在运行时我可以获得属性;但我也希望将类的静态变量作为列表.那么如何获取静态变量的名称和值列表呢?这是我的模型类的结构:解决方法尚不支持通过反射获取类型属性.如果将类对象传递给初始化程序,则可以通过检查镜像的displayStyle来查看:

  4. Swift实现类网易云音乐横向scroll menu切换菜单一

    本文准备从头开始分析该菜单控件的实现过程,也作为个人的学习笔记。首先,位于ViewController.swift中viewDidLoad函数中的self.title="PAGEMENU"navigationController?.navigationBar.tintColor=UIColor.whiteColor()//这句并没有什么作用用于配置基础页面的标题内容字体,如上图所示。varcontrollerArray:[UIViewController]=[]配置用于展示的viewcontroller,

  5. swift 2.0 类中的继承

    //class说明这个类方法是可以遗传下去的,子类也是可以对这个类进行进一步整理。

  6. NSDate格式化小例

    使用NSDate可以获到当前时间,得到的是一个NSDate时间对象,使用时,可能需要将其格式化为String类型。

  7. swift - 为类和结构体增加下标subscript操作

    数组和字典都是struct类型,而且都支持下标操作。类,也支持下标操作。

  8. Swift 类的继承 convenience required

    看下面的代码上面打印结果iamaSolider,nameissolideriamaWorker,nameisworkeriamaSolider,nameisworkerSwift的两段式构造

  9. 属性 – Swift类中的错误:属性未在super.init调用初始化

    我有两个类,Shape和Square使用上面的实现我得到错误:为什么我必须在调用super.init之前设置self.sideLength?

  10. 如何在Swift中创建类方法/属性?

    Objective-C中的类(或静态)方法是在声明中完成的。如何在Swift中实现?它们称为typeproperties和typemethods,您使用类或静态关键字。

随机推荐

  1. 基于EJB技术的商务预订系统的开发

    用EJB结构开发的应用程序是可伸缩的、事务型的、多用户安全的。总的来说,EJB是一个组件事务监控的标准服务器端的组件模型。基于EJB技术的系统结构模型EJB结构是一个服务端组件结构,是一个层次性结构,其结构模型如图1所示。图2:商务预订系统的构架EntityBean是为了现实世界的对象建造的模型,这些对象通常是数据库的一些持久记录。

  2. Java利用POI实现导入导出Excel表格

    这篇文章主要为大家详细介绍了Java利用POI实现导入导出Excel表格,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  3. Mybatis分页插件PageHelper手写实现示例

    这篇文章主要为大家介绍了Mybatis分页插件PageHelper手写实现示例,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

  4. (jsp/html)网页上嵌入播放器(常用播放器代码整理)

    网页上嵌入播放器,只要在HTML上添加以上代码就OK了,下面整理了一些常用的播放器代码,总有一款适合你,感兴趣的朋友可以参考下哈,希望对你有所帮助

  5. Java 阻塞队列BlockingQueue详解

    本文详细介绍了BlockingQueue家庭中的所有成员,包括他们各自的功能以及常见使用场景,通过实例代码介绍了Java 阻塞队列BlockingQueue的相关知识,需要的朋友可以参考下

  6. Java异常Exception详细讲解

    异常就是不正常,比如当我们身体出现了异常我们会根据身体情况选择喝开水、吃药、看病、等 异常处理方法。 java异常处理机制是我们java语言使用异常处理机制为程序提供了错误处理的能力,程序出现的错误,程序可以安全的退出,以保证程序正常的运行等

  7. Java Bean 作用域及它的几种类型介绍

    这篇文章主要介绍了Java Bean作用域及它的几种类型介绍,Spring框架作为一个管理Bean的IoC容器,那么Bean自然是Spring中的重要资源了,那Bean的作用域又是什么,接下来我们一起进入文章详细学习吧

  8. 面试突击之跨域问题的解决方案详解

    跨域问题本质是浏览器的一种保护机制,它的初衷是为了保证用户的安全,防止恶意网站窃取数据。那怎么解决这个问题呢?接下来我们一起来看

  9. Mybatis-Plus接口BaseMapper与Services使用详解

    这篇文章主要为大家介绍了Mybatis-Plus接口BaseMapper与Services使用详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

  10. mybatis-plus雪花算法增强idworker的实现

    今天聊聊在mybatis-plus中引入分布式ID生成框架idworker,进一步增强实现生成分布式唯一ID,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

返回
顶部