函数式接口、匿名实现类、Lambda表达式之间的关系

函数式接口

函数式接口(Functional Interface)指的是有且仅有一个抽象方法的接口。

可通过@FunctionalInterface注解将某个接口声明为函数式接口:

@FunctionalInterface
interface GreetingService 
{
    void sayMessage(String message);
}

常见的函数式接口有:

  • 消费型接口 Consumer<T>
  • 供给型接口 Supplier<T>
  • 函数型接口 Function<T,R>
  • 断言型接口 Predicate<T>

详细介绍和应用场景可见:链接

匿名实现类

对于上述接口GreetingService方法最朴素的实现方法是创建一个实现类,然后在实现类中对接口抽象方法进行实现。

//第一步:创建实现类
public class GreetingServiceImp implements GreetingService{  
  
    @Override  
    public void sayMessage(String message) {  
        System.out.println(message);  
    }  
}
//第二步:调用实现的方法
public class SayHello {  
    public static void main(String[] args) {  
        GreetingServiceImp greetingServiceImp = new GreetingServiceImp();  
        greetingServiceImp.sayMessage("Hello!");  
    }  
}

但对于只需要临时使用的方法,这种方法就显得过于麻烦了。更简单的办法是使用匿名类:

public class SayHello {  
    public static void main(String[] args) {  
        GreetingService greetingService = new GreetingService() {  
            @Override  
            public void sayMessage(String message) {  
                System.out.println(message);  
            }  
        };  
        greetingService.sayMessage("Hello!");  
    }  
}

Lambda表达式

在Java8后加入了lambda表达式,可以进一步简化对接口中方法的实现。

不过需要注意的是,lambda表达式只适用于对函数式接口进行实现。

lambda表达式的基本形式为:

//左边为输入参数,右边为lambda体
(parameters) -> {statements;} 

以下是lambda表达式的重要特征:

  • 类型声明(可选):不需要声明参数类型,编译器也可以统一识别参数值。
  • 参数圆括号(可选):一个参数无需定义圆括号,但多个参数需要定义圆括号。
  • 大括号(可选):如果主体包含了一个语句,就不需要使用大括号。
  • return关键字(可选):如果主体只有一个表达式返回值则编译器会自动返回值,大括号需要指定表达式返回了一个数值。

下面用lambda表达式来实现GreetingService接口的sayMessage方法。

//通过匿名类来实现sayMessage方法
GreetingService greetingService = new GreetingService() {  
            @Override  
            public void sayMessage(String message) {  
                System.out.println(message);  
            }  
        };  
greetingService.sayMessage("Hello!"); 

//通过lambda表达式来实现sayMessage方法,返回的是一个greetingService对象。
GreetingService greetingService = message -> System.out.println(message);  
greetingService.sayMessage("Hello!");

可以看到,相对于通过匿名类来实现接口方法,lambda表达式进一步简化了代码的编写。

重写接口方法的代码编写复杂度对比:创建具体的接口实现类 > 匿名类 > lambda 表达式

此外,如果你熟悉常用函数式接口的话,会发现我们上面做的东西其实就是消费一个参数,却不返回任何东西(这里把返回void类型当做无返回)。因此Java8内置的函数式接口Consumer<T>即可实现对应功能,不需要额外去定义一个GreetingService接口。

public static void main(String[] args) {  
    Consumer<String> consumer = s -> System.out.println(s);  
    consumer.accept("Hello!");  
}

总结

通过匿名实现类可以对接口中的抽象方法进行快速实现。

进一步地,如果该接口是函数式接口(即只有一个抽象方法,其他方法可以有多个),那么可以用lambda表达式来进一步简化代码。

ps:在某些场景下使用「方法引用」代替lambda表达式可以进一步简化代码,这在日常开发使用StreamAPI时十分常用,不过通过IDEA的Alt+Enter组合键可以很方便地实现lambda表达式和方法引用的转换,因此也不必过多纠结于方法引用,能看懂就行。

  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值