详细解读Java的Lambda表达式

作者:CrazyDragon_King 时间:2021-12-30 15:32:36 

Lambda 表达式

最早接触到 Lambda 表达式的时候,是在学习 python 的时候,当时就很好奇。后来,才发现 Java 也有这个方面的知识,最近看了相关的知识,特定来总结一下。

Lambada 简介

lambda 表达式 是Java 8新加入的新特性,它在Java中是引入了函数式编程这一概念。那么什么是函数式编程呢?

函数式编程:函数式编程是面向数学的抽象,将计算描述为一种表达式求值。

我们平常所说的面向对象编程属于命令式编程,函数式编程和命令式编程的区别是:

  • 函数式编程关心数据的映射,命令式编程关系解决问题的步骤。

  • 函数式编程关系类型(代数结构)之间的关系,命令式编程关系解决问题的步骤。

函数式编程的本质:

函数式编程中的函数指的不是计算机中的函数,而是数学中的函数,即自变量的映射。即:一个函数的值仅取决于函数参数的值,不依赖其他状态。

严格意义上的函数式编程意味着不使用可变的变量,赋值,循环和其他命令式控制结构进行编程。

函数式编程的好处:

函数式编程的好处是主要是不可变性带来的。没有可变的状态,函数就是引用透明(Referential transparency)的和没有副作用的(No Side Effect)。

上面这些都是一些基本的概念,但是我们平时可能接触这些方面的东西比较少,所以一开始感觉函数式编程是很难得东西。

简单的示例

Talk is cheap, show me the code!

先来一个最简单的例子,可能也是介绍的最多的例子了。哈哈!

给按钮添加监视器。

使用匿名内部类的方式,进行添加。

submit.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
JOptionPane.showMessageDialog(null, "点击了确定按钮", "确定", JOptionPane.INFORMATION_MESSAGE);
}
});

这种方式的缺点:使用了很多的模板代码,真正必要的代码,只是方法体内的代码。所以,Java 8 引入的 Lambda 表达式可以简化这种代码(当然了,也是有限制的,不是所有的匿名内部类都可以,这个后面会提到。)。

使用 Lambda 表达式 简化代码

submit.addActionListener((e)->{
JOptionPane.showMessageDialog(null, "点击了确定按钮", "确定", JOptionPane.INFORMATION_MESSAGE);
});

Lambda 表达式是一个匿名方法,将行为像数据一样进行传递。

说明

可以看出来,使用 Lambda 表达式简化后的代码表达变得更加清晰了,并且不用再去写繁琐的模板代码了。

进一步简化

参数括号和代码体的花括号也可以省略(只有一个参数时,可以省略圆括号,只有一行代码时,可以省略花括号)。

ActionListener listener = e->JOptionPane.showMessageDialog(null, "点击了确定按钮", "确定", JOptionPane.INFORMATION_MESSAGE);

小结
当使用 Lambda 表达式代替匿名内部类创建对象时,Lambda 表达式的代码块将会替代实现抽象方法的方法体,Lambda 就相当于一个匿名方法。

Lambda 表达式的组成部分

lambda 表达式由三部分组成:

  • 形参列表。形参列表允许省略形参类型。如果形参列表中只有一个参数,可以省略形参列表的圆括号。

  • 箭头(->)。英文短线和大于号。

  • 代码块。如果代码块只有一句,可以省略花括号。如果只有一条 return 语句,可以省略 return,lambda表达式会自动返回这条语句的值。

注:
之所以可以省略形参列表是因为 编译器 可以进行类型推断,例如:

List<Dog> dogs1 = new ArrayList<Dog>();

List<Dog> dogs2 = new ArrayList<>();

上面使用 菱形语法,可以省略尖括号里面的东西,这就是类型推断的作用
但是类型推断也不是万能的,不是所有的都可以推断出来的,所以有时候,还是要显示的添加形参类型,例如:

先不要管这个代码的具体作用。

BinaryOperator b = (x, y)->x*y;
//上面这句代码无法通过编译,下面是报错信息:无法将 * 运算符作用于 java.lang.Object 类型。
The operator * is undefined for the argument type(s) java.lang.Object, java.lang.Object
//添加参数类型,正确的代码。
BinaryOperator<Integer> b = (x, y)->x*y;

所以,类型推断不是万能的,如果编译器无法推断,那就是我们的错误,不要过度依赖编译器。有时候,显示的添加参数类型,还是很必要的,当然了,这需要去多练习。

函数式接口

前面了解了,Lambda 表达式可以代替匿名内部类,进而达到简化代码,表达清晰的目的。那么使用 Lambda 表示式的前提是什么呢?-- 函数式接口

Lambda 表达式的类型,也被称为 目标类型 (Target Type),它必须是一个函数式接口(Functional Interface)。所谓函数式接口,指的就是:只包含一个抽象方法的接口。(可以包含多个默认方法,静态方法,但必须只有一个抽象方法)。
注:Java 8 专门提供了一个注解:@FunctionalInterface。用于标注某个接口是函数式接口,这样编译时就会检查,如果该接口含有多个抽象方法,编译器就会报错。

上面使用 Lambda 表达式来为按钮添加了监视器,可以看出来,Lambda 表达式 代替了 new ActionListener()对象。

所以 Lambda 的表达式就是被当成一个对象。

例如:

ActionListener listener = e->JOptionPane.showMessageDialog(null, "点击了确定按钮", "确定", JOptionPane.INFORMATION_MESSAGE);

从上面这个例子中可以看出来,Lambda 表达式实现的是匿名方法&ndash;因此它只能实现特定函数式接口中的唯一方法。
所以 Lambda 表达式有下面两种限制:

Lambda 表达式的目标类型必须是明确的函数式接口。 Lambda 表达式只能为函数式接口创建对象。Lambda只能实现一个方法,因此它只能为含有一个抽象方法的接口(函数式接口)创建对象。

介绍几个 Java 中重要的函数接口

详细解读Java的Lambda表达式

从这种表可以看出来,抽象方法的名字反而不是最重要的了,重要的是参数和返回值。 因为在写 Lambda 表达式的时候,也不要使用 抽象方法名了。

上面使用几个简单的例子来说明上面接口的应用:

测试代码

import java.text.ParseException;
import java.util.function.BinaryOperator;
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.function.Predicate;
import java.util.function.Supplier;
import java.util.function.UnaryOperator;
import java.util.stream.Stream;

public class Test {
public static void main(String[] args) throws ParseException {
//Lambda 表达式中的构造器引用,用于简化代码。
Creat<Dog> c = Dog::new;
Dog dog = c.creat("小黑", 15);
System.out.println(dog.toString());

Predicate<String> predicate = (words)->{
return words.length() > 20;
};
assert predicate.test("I love you yesterday and today!") : "长度小于20";
assert !predicate.test("God bless you!") : "长度小于20";
System.out.println("------------------------");

Consumer<Dog> consumer = System.out::println;
consumer.accept(dog);
System.out.println("------------------------");

Function<Dog, String> function = (dogObj)->{
return dogObj.getName();
};
System.out.println(function.apply(dog));
System.out.println("------------------------");

Supplier<Dog> supplier = ()->{
return new Dog("大黄", 4);
};
System.out.println(supplier.get());

//一元操作符
UnaryOperator<Boolean> unaryOperation = (flag)->{
return !flag;
};
System.out.println(unaryOperation.apply(true));

BinaryOperator<Integer> binaryOperator = (x, y)->x*y;
int result = binaryOperator.apply(999, 9999);
System.out.println(result);
}
}

测试使用的实体类

public class Dog {
private String name;
private int age;

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

public String getName() {
return name;
}

public void setName(String name) {
this.name = name;
}

public int getAge() {
return age;
}

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

@Override
public String toString() {
return "Dog [name=" + name + ", age=" + age + "]";
}
}

自定义函数式接口

@FunctionalInterface
public interface Creat<T> {
public T creat(String name, int age);
}

运行截图就不放了,感兴趣的可以试一下。

说明
我这里直接使用 Lambda 创建了对象,然后调用了这个对象的方法(就是lambda 的代码块部分),真正使用的时候,都是直接传递 Lambda 表达式的,这种方法并不推荐,但是可以让我们很好的理解为什么? 可以看出来,Lambda 表达式的作用,最后还是需要调用 重写的抽象方法的,只不过使用表达更加清晰,简化了代码。

例如:

List<Dog> dogs = new ArrayList<>();
dogs.add(new Dog("大黄", 2));
dogs.add(new Dog("小黑", 3));
dogs.add(new Dog("小哈",1));
//将行为像数据一样传递,使用集合的 forEach 方法来遍历集合,
//参数可以是一个 Lambda 表达式。
Consumer<? super Dog> con = (e)->{
System.out.println(e);
};
dogs.forEach(con);
System.out.println("--------------------------\n");

//直接传递 Lambda 表达式,更加简洁
dogs.forEach(e->System.out.println(e));
System.out.println("--------------------------\n");

//使用方法引用,进一步简化(可以看我的另一篇关于方法引用的博客)
dogs.forEach(System.out::println);
System.out.println("--------------------------\n");

//使用 Lambda 对集合进行定制排序,按照年龄排序(从小到大)。
dogs.sort((e1, e2)->e1.getAge()-e2.getAge());
dogs.forEach(System.out::println);

可以看出来,通过使用 Lambda 表达式可以,极大的简化代码,更加方便的操作集合。值得一提的是:Lambda 表达式 和 Stream 的结合,可以拥有更加丰富的操作,这也是下一步学习的方向。

运行截图:

详细解读Java的Lambda表达式

最后说一下

这个博客介绍了一些 Lambda 表达式的基本知识,但是描述的还是不是很清楚,哈哈!但是,希望通过这个简单的博客,可以帮助了解一些关于 Lambda 表达式的基本知识,这些都是很简单的东西,太难的部分,我可能暂时还没有触及到。

Lambda 表达式是一个匿名方法,将行为像数据一样进行传递。
这句话很关键,这是对 Lambda 的一个很好的总结。

来源:https://blog.csdn.net/qq_40734247/article/details/103524922

标签:Java,Lambda,表达式
0
投稿

猜你喜欢

  • C#用户定义类型转换详解

    2022-06-07 11:44:32
  • Java执行JS脚本工具

    2022-04-07 08:00:16
  • Java内存模型与JVM运行时数据区的区别详解

    2023-11-24 13:29:08
  • SpringCloud分布式链路跟踪的方法

    2023-11-24 23:42:19
  • Spring注解@Configuration与@Bean注册组件的使用详解

    2022-09-13 01:52:56
  • java中TESTful架构原理分析

    2022-03-02 21:12:10
  • C# 系统热键注册实现代码

    2021-10-07 00:12:14
  • Java实现JDK动态代理的原理详解

    2021-09-19 08:17:43
  • SpringBoot2 参数管理实践之入参出参与校验的方式

    2022-12-25 02:44:30
  • Java使用JDBC或MyBatis框架向Oracle中插入XMLType数据

    2023-10-21 04:12:22
  • SpringBoot 整合 Shiro 密码登录的实现代码

    2023-11-10 11:27:59
  • Java实现添加,读取和删除Excel图片的方法详解

    2023-11-27 06:29:33
  • 谈谈Java中整数类型(short int long)的存储方式

    2023-01-01 08:24:33
  • java 文件上传到读取文件内容的实例

    2023-11-09 22:00:27
  • 从搭建Struts2 开发环境说起

    2023-11-18 08:54:53
  • Android实现摇一摇功能

    2023-07-23 20:21:11
  • JFinal使用ajaxfileupload实现图片上传及预览

    2023-08-05 08:30:48
  • MyBatis映射文件resultMap元素中使用多个association的方法

    2023-11-29 06:53:51
  • iOS中的导航栏UINavigationBar与工具栏UIToolBar要点解析

    2023-07-08 16:52:22
  • Java 导出excel进行换行的案例

    2021-07-29 04:09:36
  • asp之家 软件编程 m.aspxhome.com