鹰击长空,鱼翔浅底,万类霜天竞自由

Java 8的lambda表达式与方法重载问题


笔者之前在学习Java8新特性的时候,最吸引我的就是lambda表达式,它无疑为Java函数编程提供了强有力的支持。lambda表达式的使用方法很简单,下面给出最简单的用法。

// InterfaceA.java 
// 在这里声明了一个函数式接口
@FunctionalInterface
public interface InterfaceA {
    void function(String a);
}
// LambdaDemo.java 
public class LambdaDemo {
    public static void main(String[] args) {
        // 使用lambda表达式创建一个匿名对象,打印参数
        func(param->System.out.println(param));
    }
    public static void func(InterfaceB interfaceB) {
        interface2.function("Hello, world!");
    }
}

lambda表达式的相关知识不再赘述,函数式接口在使用的时候无需指定参数类型,后面直接跟->和Java代码即可。因为编译器会根据函数式接口InterfaceA的方法function(Integer a)自动确定参数类型。那么这样就会导致一个问题(脑洞大开的我自然想到这个问题),万一参数确定不了怎么办。那么在什么情况下确定不了呢?对,方法重载!

// InterfaceA.java
// 在这里声明了一个函数式接口
@FunctionalInterface
public interface InterfaceA {
    void function(String a);
}
// InterfaceB.java
@FunctionalInterface
public interface InterfaceB {
    void function(Integer a);
}
// LambdaDemo.java
public class LambdaDemo {
    public static void main(String[] args) {
        func(param->System.out.println(param));
    }
    public static void func(InterfaceB interfaceB) {
        interfaceA.function(0);
    }
    public static void func(InterfaceA interfaceA) {
        interfaceB.function("Hello, world!");
    }
}

上面的代码func(param->System.out.println(param));会在编译中产生错误,因为编译器根本无法确定param到底是什么类型,这不就很蛋疼了么?确实蛋疼!不过肯定不会束手无策的,用下面的方法就可以轻松解决问题。

public static void main(String[] args) {
    func((String param)->System.out.println(param));
}

也就是在参数param前面加上String以告诉编译器它是字符串类型,这样编译器就会确定调用的实际方法是public static void func(InterfaceA interfaceA);这下问题便迎刃而解。

这个疑问是去年在学习Java8中产生的,今天无聊用IDE试了一下确实是这样的,就将该疑问以及解决办法记录下来。

Eclipse提示的信息是:The method func(InterfaceA) is ambiguous for the type LambdaDemo.


版权声明

本作品采用知识共享署名-非商业性使用 4.0 国际许可协议进行许可。 本站博文除注明转载/出处外,均为本站原创或翻译,转载前请务必署名。