目录
前言
函数式编程作为Java8最重要的新特性之一,一定程度上提升了使用Java编码的体验,我们平时多多少少也都会使用到函数式接口,但可能缺乏系统性的认知。
比如Lambda表达式的定义,Stream API的使用,Optional的定义和作用,他们跟函数式接口的联系,System.out::println这类语法的产生。
函数式编程
函数式编程是一种编程方式、编程风格,即针对确定的输入,有确定的输出。
函数式编程注重数据的映射,一个函数的返回值,仅仅依赖于参数的值,而不会因为其他外部的状态不同而不同。
函数式编程例子
public class Example {
public static void main(String[] args) {
List<String> list = Arrays.asList("Lambda","operator","calculate","SpringBoot","Java");
//打印首字母为大写的字符串
list.stream().filter(str -> Character.isUpperCase(str.charAt(0)))
.forEach(System.out::println);
//等同于
for(String str : list){
if(Character.isUpperCase(str.charAt(0))){
System.out.println(str);
}
}
//打印每个字符串首个字母
list.stream().map(str -> str.substring(0,1))
.forEach(System.out::println);
//等同于
for(String str : list){
System.out.println(str.substring(0,1));
}
}
}
理解不了没关系,慢慢往下看
函数式接口(Functional Interface)
在java.util.function包中定义了43个函数式接口,他们都有共同的特点:
- 一个函数式接口只有一个抽象方法(single abstract method);
- Object 类中的public abstract method 不会被视为单一的抽象方法;
- 函数式接口可以有默认方法和静态方法;
- 函数式接口可以用@FunctionalInterface注解进行修饰;
满足这些条件的interface,就可以被视为函数式接口;函数式接口可以不用@FunctionalInterface修饰,但用@FunctionalInterface修饰的接口一定是函数式接口。
例如Java8中的Comparator接口就是函数式接口,其部分代码:
@FunctionalInterface
public interface Comparator<T> {
//唯一的抽象方法
int compare(T o1,T o2);
//Object类的public abstrac method
boolean equals(Object obj);
//默认方法
default Comparator<T> reversed(){
return Collections.reverseOrder(this);
}
//静态方法
public static <T extends Comparable<? super T>> Comparator<T> reverseOrder(){
return Collections.reverseOrder();
}
}
Java8中的函数式接口,主要分为Predicate 接口、Function 接口、Consumer 接口、Supplier 接口;有兴趣可以去逐个看一下。
Lambda表达式
Lambda表达式本质是声明了一个函数式接口的匿名实现类;所以Lambda表达式是为了简化函数式接口实现而创建的语法糖。
Lambda表达式的语法如下:
基本结构:(args ...) -> {body}
参数可以是零个或多个,空括号用于表示一组空的参数;
参数用小括号括起来,逗号分隔,可以显式声明参数类型,也可以有编译器自动从上下文推断参数的类型。例如(a,b)或(int a,int b);
当有且仅有一个参数时,如果不显示指明类型,则不必使用小括号;例如 a -> a*a;
body部分可以包含零条、一条或多条语句,如果body部分只有一条语句,则大括号可不用写,return和分号也可以省略,但是return类型要与匿名类的重写方法返回类型一致;如果body部分有一条以上的语句,必须包含在大括号(代码块)中,并且每个语句需要有分号,treturn类型要与匿名类的重写方法返回类型一致。