Lambda

Lambda(匿名函数)作为新特性在Java8中引入。其中涉及的概念有:行为参数化、匿名类、Lambda表达式和方法引用。

一、表达式

1.1 Lambda表达式格式:

Lambda表达式
语法的特点:
* 参数可为空,参数可通过前文进行类型推断,因此可以不加参数类型。
* 主体可以为表达式或者语句。
* 语句需要加{},可多行语句。

1.2 有效的Lambda表达式

有效的Lambda表达式

1.3 Lambda示例

Lambda示例

二、函数式接口与函数式描述符

2.1 函数式接口

函数式接口:只定义了一个抽象方法的接口(如:Comparator和Runnable)。可加@FunctionalInterface,该注解可帮助编译器进行检查,不必须。

2.1.1 函数式接口示例

函数式接口示例

Lambda表达式允许直接以内联的形式为函数式接口的抽象方法提供实现,把整个表达式作为函数式接口的实例(函数式接口的一个具体实现的实例)。

例子如下,下面的无参数有效,因为Runnable是一个只定义了一个抽象方法run的函数式接口:
[图片上传失败...(image-b6218b-1586231371299)]
images.jianshu.io/upload_images/12234310-1e96a98eea0e1cb3.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

2.2 函数描述符

函数描述符:函数式接口的抽象方法的签名基本上就是Lambda表达式的签名,我们将这种抽象方法称之为函数描述符。

2.2.1 函数描述符示例

//代表了参数列表为空,且无返回值的函数,其正是Runnable接口的代表
() -> void

public void process(Runnable r){
    r.run();
}

//Lambda表达式如下,不接受参数且返回void
process(() -> System.out.println("lambda表达式!!"))

三、常用的函数式接口

函数式接口可以理解为对行为动作的抽象,常用的有Consumer、Supplier、Predicate、Function接口,是对常用动作的进一步总结。
  • Consumer(消费型接口,可用于读取数据)

    @FunctionalInterface
    public interface Consumer<T> {
        //有参数,没有返回值
        void accept(T t);
    }
    
  • Supplier(供给型接口,可用于获取数据)

    @FunctionalInterface
    public interface Supplier<T> {
        //没有参数,有返回值
        T get();
    }
    
  • Predicate(谓词型接口,可用于筛选数据)

    @FunctionalInterface
    public interface Predicate<T> {
        //有参数,返回布尔值
        boolean test(T t);
    }
    
  • Function(功能型接口,可用于转换数据)

    @FunctionalInterface
    public interface Function<T, R> {
        R apply(T t);
    }
    

四、方法引用

如果一个Lambda代表的只是【直接调用这个方法】,最好使用名称来调用,而不是使用描述调用它。
方法引用示例

注意:

  1. 如果Lambda表达式抛出一个异常,那么抽象方法所声明的throws语句必须与之匹配。
  2. Lambda表达式支持类型推断,因此声明参数时可省略类型。
//没有类型推断
Comparator<Apple> c = (Apple a1, Apple a2) -> a1.getWeight().compareTo(a2.getWeight());
//有类型推断
Comparator<Apple> c = (a1, a2) -> a1.getWeight().compareTo(a2.getWeight());
  1. Lamdba表达式引用的局部变量必须是最终的(final)或事实上最终的。
//以下代码将无法编译
int portNumber = 1337;
Runnable r = () -> System.out.println(portNumber);
portNumber = 31337;
//
//
对局部变量限制的原因(详情请参考Java8实战 P52):
* 实例变量和局部变量背后的实现不同,实例变量存储在堆中,而局部变量则保存在栈上。
* 不鼓励使用改变外部变量的典型命令式编程模式,这种模式会阻碍并行处理。

Lambda表达式使用的经典案例

//实现一个List的排序
//sort方法的签名为:void sort(Somparator<? super E> c)

//1. 传递代码
public class AppleComparator implements Comparator<Apple>{
    public int compare(Apple a1, Apple a2){
        return a1.getWeight.compareTo(a2.getWeight());
    }
}
inventory.sort(new AppleComparator());   

//2. 使用匿名类
inventory.sort(new Comparator<Apple>() {
    public int compare(Apple a1, Apple a2){
        return a1.getWeight().compareTo(a2.getWeight());
    }
}) 

//3. 使用Lambda表达式
inventory.sort((Apple a1, Apple a2) -> a1.getWight().compareTo(a2.getWeight()));
//使用类型推断
inventory.sort(a1, a2) -> a1.getWight().compareTo(a2.getWeight()));
//使用Comparator中的comparing静态辅助方法
inventory.sort(comparing(a -> a.getWeight()));

//4. 使用方法引用
inventory.sort(comparing(Apple::getWeight));
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 简介 概念 Lambda 表达式可以理解为简洁地表示可传递的匿名函数的一种方式:它没有名称,但它有参数列表、函数主...
    刘涤生阅读 8,430评论 5 18
  • Java Lambda概要 Java Lambda表达式是一种匿名函数;它没有声明的方法,即没有访问修饰符、返回值...
    小波同学阅读 5,916评论 0 47
  • 注:之前关于Java8的认知一直停留在知道有哪些修改和新的API上,对Lambda的认识也是仅仅限于对匿名内部类的...
    mualex阅读 7,819评论 1 4
  • 1、Lambda表达式 1.1 什么是Lambda 从java8出现以来Lambda,也可称为闭包(closure...
    冰河winner阅读 3,878评论 0 3
  • Java8-Lambda Java 8 的最大变化是引入了 Lambda 表达式——一种紧凑的、传递行为的方式 引...
    Cool_Pomelo阅读 2,865评论 0 5