前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >Java8中的Lambda表达式

Java8中的Lambda表达式

原创
作者头像
汤圆学Java
修改于 2021-04-19 02:08:39
修改于 2021-04-19 02:08:39
38700
代码可运行
举报
文章被收录于专栏:汤圆学Java汤圆学Java
运行总次数:0
代码可运行

作者:汤圆

个人博客:javalover.cc

前言

大家好啊,我是汤圆,今天给大家带来的是《Java8中的Lambda表达式》,希望对大家有帮助,谢谢

文章纯属原创,个人总结难免有差错,如果有,麻烦在评论区回复或后台私信,谢啦

简介

Lambda表达式是一个可传递的代码块,可以在以后执行一次或多次;

下面贴个对比代码:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
// Java8之前:旧的写法
Runnable runnable = new Runnable() {
  @Override
  public void run() {
    System.out.println("old run");
  }
};
Thread t = new Thread(runnable);// Java8之后:新的写法
Runnable runnable1 = ()->{
  System.out.println("lambda run");
};
Thread t1 = new Thread(runnable1);

可以看到,有了lambda,代码变得简洁多了

你可以把lambda当作一个语法糖

下面让我们一起来探索lambda的美好世界吧

目录

下面列出本文的目录

  • lambda的语法
  • 为啥引入lambda
  • 什么是函数式接口
  • 什么是行为参数化
  • 手写一个函数式接口
  • 常用的函数式接口
  • 什么是方法引用
  • 什么是构造引用
  • lambda的组合操作

正文

1. lambda的语法

下面分别说下语法中的三个组成部分

  • 参数: ( Dog dog )
    • 参数类型可省略(当编译器可以自动推导时),比如Comparator<String> comparatorTest = (a, b)->a.length()-b.length();,可以推导出a,b都为String
    • 当参数类型可省略,且只有一个参数时,括弧也可以省略(但是个人习惯保留)
  • 符号: ->
  • 主体:{ System.out.println("javalover"); }
    • 如果是一条语句,则需要加大括号和分号{;}(比如上图所示)
    • 如果是一个表达式,则直接写,啥也不加(比如a.length()- b.length()

2. 为啥引入lambda

为了简化代码

因为Java是面向对象语言,所以在lambda出现之前,我们需要先构造一个对象,然后在对象的方法中实现具体的内容,再把构造的对象传递给某个对象或方法

但是有了lambda以后,我们可以直接将代码块传递给对象或方法

现在再回头看下开头的例子

可以看到,用了lambda表达式后,少了很多模板代码,只剩下一个代码块(最核心的部分)

3. 什么是函数式接口

就是只定义了一个抽象方法的接口

  • 正例:有多个默认方法,但是如果只有一个抽象方法,那它就是函数式接口,示例代码如下
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
@FunctionalInterface
public interface FunctionInterfaceDemo {
    void abstractFun();
    default void fun1(){
        System.out.println("fun1");    
    }
    default void fun2(){
        System.out.println("fun2");
    }   
}

这里的注解@FunctionalInterface可以省略,但是建议加上,就是为了告诉编译器,这是一个函数式接口,此时如果该接口有多个抽象方法,那么编译器就会报错

  • 反例:比如A extends B,A和B各有一个抽象方法,那么A就不是函数式接口,示例代码如下
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
// 编译器会报错,Multiple non-overriding abstract methods found in XXX
@FunctionalInterface
public interface NoFunctionInterfaceDemo extends FunctionInterfaceDemo{
  void abstractFun2();
}

上面的父接口FunctionInterfaceDemo中已经有了一个抽象方法,此时NoFunctionInterfaceDemo又定义了一个抽象方法,结果编译器就提示了:存在多个抽象方法

在Java8之前,其实我们已经接触过函数式接口

比如Runnable 和 Comparable

只是没有注解@FunctionalInterface。

那这个函数式接口要怎么用呢?

配合lambda食用,效果最佳(就是把lambda传递给函数式接口),示例代码如下:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
new Thread(() -> System.out.println("run")).start();

其中用到的函数式接口是Runnable

4. 什么是行为参数化

就是把行为定义成参数,行为就是函数式接口

类似泛型中的类型参数化<T>,类型参数化是把类型定义成参数

行为参数化,通俗点来说:

  • 就是用函数式接口形参
  • 然后传入接口的各种实现内容(即lambda表达式)作为实参
  • 最后在lambda内实现各种行为(好像又回到多态的那一节了?这也是为啥多态是Java的三大特性的原因之一,应用太广泛了)

这样来看的话,行为参数化和设计模式中的策略模式有点像了(后面章节会分别讲常用的几种设计模式)

下面我们手写一个函数式接口来加深理解吧

5. 手写一个函数式接口

下面我们循序渐进,先从简单的需求开始

  • 第一步:比如我们想要读取某个文件,那可以有如下方法:
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public static String processFile() throws IOException {
    // Java7新增的语法,try(){},可自动关闭资源,减少了代码的臃肿
    try( BufferedReader bufferedReader = 
        new BufferedReader(new  FileReader("D:\\JavaProject\\JavaBasicDemo\\test.txt"))){
        return bufferedReader.readLine();
    }
}

可以看到,核心的行为动作就是 return bufferedReader.readLine();,表示读取第一行的数据并返回

那如果我们想要读取两行呢?三行?

  • 第二步:这时就需要用到上面的函数式接口了,下面就是我们自己编写的函数式接口
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
@FunctionalInterface
interface FileReadInterface{
    // 这里接受一个BufferedReader对象,返回一个String对象
    String process(BufferedReader reader) throws IOException;
}

可以看到,只有一个抽象方法process(),它就是用来处理第一步中的核心动作(读取文件内容)

至于想读取多少内容,那就需要我们在lambda表达式中定义了

  • 第三步:接下来我们定义多个lambda表达式,用来传递函数式接口,其中每个lambda表达式就代表了一种不同的行为,代码如下:
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
// 读取一行
FileReadInterface fileReadInterface = reader -> reader.readLine();
// 读取两行
FileReadInterface fileReadInterface2 = reader -> reader.readLine() + reader.readLine();
  • 第四步:我们需要修改第一步的processFile(),让其接受一个函数式接口,并调用其中的抽象方法,代码如下:
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
// 参数为第二步我们自己手写的函数式接口
public static String processFile(FileReadInterface fileReadInterface) throws IOException {
        try( BufferedReader bufferedReader =
                 new BufferedReader(new FileReader("./test.txt"))){
                    // 这里我们不再自己定义行为,而是交给函数式接口的抽象方法来处理,然后通过lambda表达式的传入来实现多个行为
          return fileReadInterface.process(bufferedReader);
        }
    }
  • 第五步:拼接后,完整代码如下:
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public class FileReaderDemo {
    public static void main(String[] args) throws IOException {
				// 第三步: 
      	// lambda表达式1 传给 函数式接口:只读取一行
      	FileReadInterface fileReadInterface = reader -> reader.readLine();
				// lambda表达式2 传给 函数式接口:只读取两行
      	FileReadInterface fileReadInterface2 = reader -> reader.readLine() + reader.readLine();
      	// 最后一步: 不同的函数式接口的实现,表现出不同的行为
        String str1 = processFile(fileReadInterface);
        String str2 = processFile(fileReadInterface2);
        System.out.println(str1);
        System.out.println(str2);
    }
  	// 第四步: 读取文件方法,接受函数式接口作为参数
    public static String processFile(FileReadInterface fileReadInterface) throws IOException {
        try( BufferedReader bufferedReader =
                 new BufferedReader(new FileReader("./test.txt"))){
					// 调用函数式接口中的抽象方法来处理数据					
          return fileReadInterface.process(bufferedReader);
        }
    }
	// 第一步:
  public static String processFile() throws IOException {
        try( BufferedReader bufferedReader =
                 new BufferedReader(new FileReader("./test.txt"))){
          return bufferReader.readLine();
        }
    }


}

// 第二步: 我们手写的函数式接口
@FunctionalInterface
interface FileReadInterface{
    String process(BufferedReader reader) throws IOException;
}

其实你会发现,我们手写的这个函数式接口,其实就是Function<T>去除泛型化后的接口,如下所示:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
@FunctionalInterface
public interface Function<T, R> {
	// 都是接受一个参数,返回另一个参数
  R apply(T t);
}

下面我们列出Java中常用的一些函数式接口,你会发现自带的已经够用了,基本不会需要我们自己去写

这里的手写只是为了自己实现一遍,可以加深理解程度

6. 常用的函数式接口

7. 什么是方法引用

我们先看一个例子

前面我们写的lambda表达式,其实还可以简化,比如

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
// 简化前
Function<Cat, Integer> function = c->c.getAge();
// 简化后
Function<Cat, Integer> function2 = Cat::getAge;

其中简化后的Cat::getAge,我们就叫做方法引用

方法引用就是引用类或对象的方法

下面我们列出方法引用的三种情况:

  1. Object::instanceMethod(对象的实例方法)
  2. Class::staticMethod(类的静态方法)
  3. Class::instanceMethod(类的实例方法)

像我们上面举的例子就是第三种:类的实例方法

下面我们用代码演示上面的三种方法:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public class ReferenceDemo {
    public static void main(String[] args) {
        // 第一种:引用对象的实例方法
        Cat cat = new Cat(1);
        Function<Cat, Integer> methodRef1 = cat::getSum; 
        // 第二种:引用类的静态方法
        Supplier<Integer> methodRef2 = Cat::getAverageAge;
        // 第三种:引用类的实例方法
        Function<Cat, Integer> methodRef3 = Cat::getAge;
    }
}
class Cat {
    int age;

    public Cat(int age) {
        this.age = age;
    }

    // 获取猫的平均年龄
    public static int getAverageAge(){
        return 15;
    }
    // 获取两只猫的年龄总和
    public int getSum(Cat cat){
        return cat.getAge() + this.getAge();
    }

    public int getAge() {
        return age;
    }    public void setAge(int age) {
        this.age = age;
    }
}

为啥要用这个方法引用呢?

方法引用好比lambda表达式的语法糖,语法更加简洁,清晰

一看就知道是调用哪个类或对象的哪个方法

8. 什么是构造引用

上面介绍了方法引用,就是直接引用某个方法

这里的构造引用同理可得,就是引用某个类的构造方法

构造引用的表达式为:Class::new,仅此一种

如果你有多个构造函数,那编译器会自己进行推断参数(你看看,多好,多简洁)

比如下面的代码:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
// 这里调用 new Cat()
Supplier<Cat> constructRef1 = Cat::new;
// 这里调用 new Cat(Integer)
Function<Integer, Cat> constructRef2 = Cat::new;

9. lambda表达式中引入外部变量的限制

要求引入lambda表达式中的变量,必须是最终变量,即该变量不会再被修改

比如下面的代码:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public static void main(String[] args) {
  String str = "javalover.cc";
  Runnable runnable = ()->{
    str = "1";// 这里会报错,因为修改了str引用的指向
    System.out.println(str);
  }
}

可以看到,lambda表达式引用了外面的str引用,但是又在表达式内部做了修改,结果就报错了

为啥要有这个限制呢?

为了线程安全,因为lambda表达式有一个好处就是只在需要的时候才会执行,而不是调用后立马执行

这样就会存在多个线程同时执行的并发问题

所以Java就从根源上解决:不让变量被修改,都是只读的

那你可能好奇,我不把str的修改代码放到表达式内部可以吗?

也不行,道理是一样的,只要lambda有用到这个变量,那这个变量不管是在哪里被修改,都是不允许的

不然的话,我这边先执行了一次lambda表达式,结果你就改了变量值,那我第二次执行lambda,不就乱了吗

10. lambda的组合操作

最后是lambda的必杀技:组合操作

在这里叫组合或者复合都可以

概述:组合操作就是先用一个lambda表达式,然后再在后面组合另一个lambda表达式,然后再在后面组合另另一个lambda表达式,然后。。。有点像是链式操作

学过JS的都知道Promise,里面的链式操作就和这里的组合操作很像

用过Lombok的朋友,应该很熟悉@Builder注解,其实就是构造者模式

下面我们用代码演示下组合操作:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
// 重点代码
public class ComposeDemo {
    public static void main(String[] args) {
        List<Dog> list = Arrays.asList(new Dog(1,2), new Dog(1, 1));
        // 1. 先按年龄排序(默认递增)
      	// Dog::getAge, 上面介绍的方法引用
      	// comparingInt, 是Comparator的一个静态方法,返回Comparator<T>
      	Comparator<Dog> comparableAge = Comparator.comparingInt(Dog::getAge);
        // 2. 如果有相同的年龄,则年龄相同的再按体重排序(如果年龄已经比较出大小,则下面的体重就不会再去比较)
        Comparator<Dog> comparableWeight = Comparator.comparingInt(Dog::getWeight);;
        // 3. 调用list对象的sort方法排序,参数是Comparator<? super Dog>
        list.sort(comparableAge.thenComparing(comparableWeight));
        System.out.println(list);
    }
}
// 非重点代码
class Dog{
    private int age;
    private int weight;

    public Dog(int age, int weight) {
        this.age = age;
        this.weight = weight;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    public int getWeight() {
        return weight;
    }

    public void setWeight(int weight) {
        this.weight = weight;
    }

    @Override
    public String toString() {
        return "Dog{" +
                "age=" + age +
                ", weight=" + weight +
                '}';
    }
}

输出:[Dog{age=1, weight=1}, Dog{age=1, weight=2}]

比较的流程如下所示:

总结

  1. lambda的语法: 参数+符合+表达式或语句,比如(a,b)->{System.out.println("javalover.cc");}
  2. 函数式接口:只有一个抽象方法,最好加@FunctionalInterface,这样编译器可及时发现错误,javadoc也说明这是一个函数式接口(可读性)
  3. 行为参数化:就是函数式接口作为参数,然后再将lambda表达式传给函数式接口,通过不同的lambda内容实现不同的行为
  4. 方法引用:lambda的语法糖,总共有三种:
    • Object::instanceMethod(对象的实例方法)
    • Class::staticMethod(类的静态方法)
    • Class::instanceMethod(类的实例方法)
  5. 构造引用:就一种,编译器自己可判断是哪个构造函数,语法为Class::new
  6. 在lambda中引入外部变量,必须保证这个变量是最终变量,即不再被修改
  7. lambda的组合操作,就是链式操作,组合是通过函数式接口的静态方法来组合(静态方法会返回另一个函数式接口的对象)

比如list.sort(comparableAge.thenComparing(comparableWeight));

后记

最后,感谢大家的观看,谢谢

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
java8实战读书笔记:Lambda表达式语法与函数式编程接口
测试:如下语句是否是正确的lambda表达式。 (1) () -> {} (2) () -> "Raoul" (3) () -> {return "Mario";} (4) (Integer i) -> return "Alan" + i; (5) (String s) -> {"IronMan";}
丁威
2019/06/11
5560
java8实战读书笔记:Lambda表达式语法与函数式编程接口
Java8学习(3)- Lambda 表达式
猪脚:以下内容参考《Java 8 in Action》 本次学习内容: Lambda 基本模式 环绕执行模式 函数式接口,类型推断 方法引用 Lambda 复合 代码: https://github.com/Ryan-Miao/someTest/blob/master/src/main/java/com/test/java8/c3/AppleSort.java 上一篇: Java8学习(2)- 通过行为参数化传递代码--lambda代替策略模式 ---- 1. 结构 初始化一个比较器: Comparato
Ryan-Miao
2018/03/13
1.1K0
Java8学习(3)- Lambda 表达式
Java8之熟透Lambda表达式
​ Lambda 表达式可以理解为简洁地表示可传递的匿名函数的一种方式:它没有名称,但它有参数列表、函数主体、返回类型,可能还有一个可以抛出的异常列表。
用户1195962
2019/09/29
5910
Java8之熟透Lambda表达式
java8新特性(一):Lambda表达式
Lambda 是一个匿名函数,我们可以把Lambda表达式理解为是一段可以传递的代码(将代码像数据一样进行传递)。可以写出更简洁、更灵活的代码。作为一种更紧凑的代码风格,使Java的语言表达能力得到了提升。
周三不加班
2019/09/03
4120
java8新特性(一):Lambda表达式
Java8新特性----Lambda表达式详细探讨
​ Lambda是一个匿名函数,可以理解为一段可以传递的代码(将代码像数据一样传递);可以写出更简洁、更灵活的代码;作为一种更紧凑的代码风格,是Java语言表达能力得到提升
大忽悠爱学习
2021/11/15
2940
一文看懂 Java8 的 Lambda表达式!
IT领域的技术日新月异,Java14问世了,但是对于国内的许多程序员来说,连Java8都还没有真正掌握。
程序员小跃
2020/04/10
4660
Java8 Lambda表达式详解手册及实例「建议收藏」
先贩卖一下焦虑,Java8发于2014年3月18日,距离现在已经快6年了,如果你对Java8的新特性还没有应用,甚至还一无所知,那你真得关注公众号“程序新视界”,好好系列的学习一下Java8的新特性。Lambda表达式已经在新框架中普通使用了,如果你对Lambda还一无所知,真得认真学习一下本篇文章了。
全栈程序员站长
2022/09/08
1K0
Java8 Lambda表达式与Stream API (一):Lambda表达式你要知道的Java8 匿名内部类、函数式接口、lambda表达式与Stream API都在这里
你要知道的Java8 匿名内部类、函数式接口、lambda表达式与Stream API都在这里 转载请注明出处 https://cloud.tencent.com/developer/user/1605429 本文主要讲解Java8 Stream API,但是要讲解这一部分需要匿名内部类、lambda表达式以及函数式接口的相关知识,本文将分为两篇文章来讲解上述内容,读者可以按需查阅。 Java 匿名内部类、lambda表达式与函数式接口 Java Stream API 本文是该系列博文的第一篇Java 匿名
WWWWDotPNG
2018/04/10
1.1K0
Lambda表达式
Lambda表达式是可以在函数式接口上使用的。函数式接口就是只定义一个抽象方法的接口。比如:
后端码匠
2019/09/02
6430
Java中lambda表达式详解
上面的代码中,e是一个lambda的对象,根据java的继承的特性,我们可以说e对象的类型是继承自eat接口。而e1是一个正常的匿名类的对象.
付威
2018/12/05
4.8K0
Java中lambda表达式详解
JAVA8之lambda表达式详解
Lambda 表达式 − Lambda允许把函数作为一个方法的参数(函数作为参数传递进方法中)
哲洛不闹
2018/09/14
1.1K0
JAVA8之lambda表达式详解
JAVA8 Lambda表达式(下)
体会到Lambda表达式的优雅与简洁,进一步记录下一步操作,
疯狂的KK
2019/08/16
5390
JAVA8 Lambda表达式(下)
Java 8——Lambda表达式
本文内容大部分来自《Java 8实战》一书 前言 在上一篇文章中,我们了解了利用行为参数化来传递代码有助于应对不断变化的需求,它允许你定义一个代码块来表示一个行为,然后传递它。一般来说,利用这个概念,你就可以编写更为灵活且可重复使用的代码了。 但是你同时也看到,使用匿名类来表示不同的行为并不令人满意:代码十分啰嗦,这会影响程序员在时间中使用行为参数化的积极性。Lambda表达式很好的解决了这个问题,它可以让你很简洁地表示一个行为或传递代码。现在你可以把Lambda表达式看作匿名功能,它基本上就是没有声明名
我没有三颗心脏
2018/04/26
1.1K0
Java 8——Lambda表达式
JDK1.8新特性之Lambda表达式
Java8中引入了一个新的操作符“ -> ”,该操作符被称为箭头操作符或Lambda操作符,箭头操作符将Lambda表达式拆分成两部分:
程序员波特
2024/01/19
1750
Java8 Lambda表达式教程
public int add(int x, int y) {         return x + y;     }
用户7886150
2020/12/15
4610
Lambda表达式大揭秘:轻松玩转JDK 8的函数式魔法
Lambda表达式是Java 8中引入的一个核心特性,它提供了一种简洁、灵活的方式来表示一段可以传递的代码。Lambda表达式的本质是一个匿名函数,它允许我们将行为作为方法参数,或者将代码本身作为数据来处理。
王也518
2024/04/16
2370
Java8 Lambda表达式入门
Lambda表达式的实质就是一个匿名函数。C#3.0引入了Lambda表达式,Java8也不甘示弱。Java8发布很久了,今天安装了JDK体验了Java8中的Lambda表达式。
卡尔曼和玻尔兹曼谁曼
2019/01/22
5450
Java 8之lambda表达式(一)
1.1 为什么要使用lambda表达式 "lambda 表达式"是一段可以传递的代码,因此它可以被执行一次或多次。下面让我们来下面的代码示例: class Worker implements Runnable{ public void run(){ for(int i=0;i<1000;i++){ doWork(); } } ... } 然后直接启动一个线程 Worker worker=new Worker(); new Thread(worker).st
code_horse
2018/07/02
3740
【Java8新特性】01 函数式接口和Lambda表达式你真的会了吗
Lambada表达式可以理解为:可传递的匿名函数的一种简洁表达方式。Lambda表达式没有名称,同普通方法一样有参数列表、函数主体、返回类型等;
爱笑的架构师
2020/09/24
4680
Java1.8之Lambda表达式
1、Java8的lambda表达式,通过lambda表达式可以替代我们之前写的匿名内部类来实现接口。lambda表达式本质是一个匿名函数。
别先生
2021/01/13
3640
推荐阅读
相关推荐
java8实战读书笔记:Lambda表达式语法与函数式编程接口
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验