首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >问答首页 >为什么有签名的两个方法(原语,包装器)和(原语,原语)会导致方法调用(包装器,原语)模棱两可?

为什么有签名的两个方法(原语,包装器)和(原语,原语)会导致方法调用(包装器,原语)模棱两可?
EN

Stack Overflow用户
提问于 2014-01-04 16:59:14
回答 3查看 1.3K关注 0票数 17

这只是一个练习,但我搞不懂模棱两可的地方:

代码语言:javascript
运行
复制
private static void flipFlop(String str, int i, Integer iRef) {
System.out.println(str + "ciao");
}

private static void flipFlop(String str, int i, int j) {
System.out.println(str + "hello");
}

public static void main(String[] args) {
flipFlop("hello", new Integer(4), 2004);
}

上面写着:

方法flipFlop(String,int,Integer)对于类型测试是不明确的。

我会猜到第二个参数将被打开到int中,因此第二个flipFlop方法将是选择。

EN

回答 3

Stack Overflow用户

回答已采纳

发布于 2014-01-04 17:11:20

如果您喜欢引人入胜的阅读,以下是Java语言规范的相关部分,它描述了如何解决方法。

但基本上,您的第三个参数可以解释为原语或自动装箱包装器,编译器无法确定您想要什么。这两种方法在使用JLS术语时都是“最大特定的”。

票数 14
EN

Stack Overflow用户

发布于 2014-01-04 18:05:48

好的,我已经仔细观察了JLS,我相信这应该可以消除你可能还有的疑问。

这就是最初的问题:

代码语言:javascript
运行
复制
public class Main {
    private static void flipFlop(int i, Integer iRef) {
        System.out.println("Method 1");
    }

    private static void flipFlop(int i, int j) {
        System.out.println("Method 2");
    }

    public static void main(String[] args) {
        flipFlop(new Integer(4), 2004);
    }
}

正如在另一个答案中所指出的:这失败了,因为编译器无法决定使用什么重载。

然而,你可能认为这没有任何意义。在这种情况下,编译器可以很好地决定应该使用什么方法:

代码语言:javascript
运行
复制
public class Main {
    private static void flipFlop(Integer y) {
        System.out.println("ciao");
    }

    private static void flipFlop(int j) {
        System.out.println("hello");
    }

    public static void main(String[] args) {
        flipFlop(new Integer(6));
        flipFlop(6);
    }
}

理性告诉我们,当你有值X + Y和分别采用Y + XY + Y的两种方法时,你知道XY是可互换的,这意味着后一种方法更具体。

JLS描述了这两者之间的差异。我提供了以下的整个工作流程,但重要的是:

首先,编译器将查看具有相同签名的方法,而则禁止装箱/取消装箱。在我们的第二个例子中,这不会引起任何问题,但是在我们的第一个示例中,这不会返回一个可满足的方法,因为它们都不接受Integer作为第一个参数。

如果失败,编译器将转到第二步,在该步骤中,允许装箱/取消装箱。这应该解决了第一个参数的问题,但现在导致了第二个参数的模糊性,因为现在还不清楚您是指使用int的重载还是使用Integer的重载。

这最终导致一个不明确的方法调用。

15.12.2编译时步骤2:确定方法签名

  1. 第一阶段(第15.12.2.2节)执行重载解析,而不允许装箱或取消装箱转换,也不允许使用可变的方法调用。如果在此阶段没有找到适用的方法,则处理将继续到第二阶段。
  2. 第二阶段(第15.12.2.3节)在允许装箱和取消装箱的同时执行重载解析,但仍然排除了变量方法调用的使用。如果在此阶段没有找到适用的方法,则处理将继续到第三阶段。

如果在适用性测试的三个阶段中的一个阶段已经确定了几种适用的方法,那么就选择最具体的方法,如§15.12.2.5节所规定的那样。

15.12.2.5选择最具体的方法

如果方法是可访问和可应用的,并且没有其他可应用和可访问的方法,则该方法对于方法调用来说是最大的特定的。 可能没有任何方法是最具体的,因为有两个或更多的方法具有最大的特定性。在这种情况下:

  • 如果所有最大特定的方法都有覆盖等效(§8.4.2)签名,那么:(.(决定谁被选中的一些规则。)
  • 否则,我们会说方法调用是不明确的,并且会发生编译时错误.
票数 11
EN

Stack Overflow用户

发布于 2014-01-04 17:11:27

您的第二个参数2004 (即int )也适用于Integer,因为自动拳击,这就是编译器无法决定使用哪种方法的原因。

票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/20923888

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档