Java中的Lambda表达式简介及应用
在接触Lambda表达式、了解其作用之前,首先来看一下,不用Lambda的时候我们是怎么来做事情的。
我们的需求是,创建一个动物(Animal)的列表,里面有动物的物种名,以及这种动物是否会跳,是否会游泳,并可以根据需要打印出会跳和会游泳的动物名。
首先我们创建一个Animal类:
public class Animal {
private String species;
private boolean canHop;
private boolean canSwim;
private boolean canFly;
public Animal(String speciesName, boolean hopper, boolean swimmer, boolean flyer) {
species = speciesName;
canHop = hopper;
canSwim = swimmer;
canFly = flyer;
}
public boolean canHop() {
return canHop;
}
public boolean canSwim() {
return canSwim;
}
public boolean canFly() {
return canFly;
}
public String toString() {
return species;
}
}
这个Animal类有四个私有的实例变量和三个共公有的方法。
变量
字符串型的species
,用来记录动物的物种名,布尔型的canHop
,canSwim
和canFly
,用来记录这种动物是否能跳,是否能游泳,是否会飞。
构造方法实例化对象的时候会给这些变量赋值。
方法
前三个是返回值为布尔型的canHop()
,canSwim()
和canFly()
,调用方法时返回canHop
,canSwim
和canFly
变量的值
另外还有一个重写的toString()
方法,用于在程序中识别动物的物种名。
下面这段代码创建了一个ArrayList并向List内添加了五个Animal对象,对应五种动物和各自的能力:
List<Animal> animals = new ArrayList<Animal>();
animals.add(new Animal("fish", false, true, false));
animals.add(new Animal("kangroo", true, false, false));
animals.add(new Animal("rabbit", true, false, false));
animals.add(new Animal("turtle", false, true, false));
animals.add(new Animal("bird", true, true, true));
为了判断某种动物是否会跳跃或者会游泳(会不会飞先不管),我们可以写两个方法来返回Animal类的canHop()
和canSwim()
,即获取到了canHop
和canSwim
私有变量:
public boolean checkIfHopper(Animal a) {
return a.canHop();
}
public boolean checkIfSwimmer(Animal a) {
return a.canSwim();
}
在此为了说明Lambda表达式的作用,在这里我们把这两个方法分别放到两个类中,并把方法名改为test
:
(不要问为什么起名叫test而不是Test或者TEST,也不要问为什么一定要把两个方法拆分到两个类里面,照做就是了,后面会说)
public class CheckIfHopper {
public boolean test(Animal a) {
return a.canHop();
}
}
public class CheckIfSwimmer {
public boolean test(Animal a) {
return a.canSwim();
}
}
(一个java文件里只能有一个public class,这里是为了方便所以把多个public class放在一段代码里的,下同)
为了校验这两种能力,我们分别创建了一个类,里面分别只有一个方法。即一个类校验一种能力。
按照这个思路,我们可以创建一个名为CheckTraits的接口,来让上面这些类来实现这个接口:
public interface CheckTraits {
boolean test(Animal a);
}
public class CheckIfHopper implements CheckTraits {
public boolean test(Animal a) {
return a.canHop();
}
}
public class CheckIfSwimmer implements CheckTraits {
public boolean test(Animal a) {
return a.canSwim();
}
}
CheckTraits接口有且只有一个方法boolean test(Animal a)
。该方法接受一个Animal类型的对象,并返回布尔值。
有了校验能力的方法,接下来我们只需要去按需要调用这些方法就好了。我们来写一个print方法,接受一个动物列表animals
和一个校验种类checker
作为参数,打印出列表内符合要求的动物:
private static void print(List<Animal> animals, CheckTraits checker) {
for (Animal animal : animals) {
if (checker.test(animal)) {
System.out.print(animal + " ");
}
System.out.println();
}
}
完整的TraditionalSearch.java代码如下:
import java.util.ArrayList;
import java.util.List;
import java.util.function.Predicate;
public class TraditionalSearch {
public static void main(String[] args) {
List<Animal> animals = new ArrayList<Animal>();
animals.add(new Animal("fish", false, true, false));
animals.add(new Animal("kangroo", true, false, false));
animals.add(new Animal("rabbit", true, false, false));
animals.add(new Animal("turtle", false, true, false));
animals.add(new Animal("bird", true, true, true));
print(animals, new CheckIfHopper());
System.out.println("-----");
print(animals, new CheckIfSwimmer());
}
private static void print(List<Animal> animals, CheckTraits checker) {
for (Animal animal : animals) {
if (checker.test(animal)) {
System.out.print(animal + " ");
}
System.out.println();
}
}
}
代码输出结果为:
kangroo
rabbit
bird
-----
fish
turtle
bird
到目前为止,我们
- 创建了一个Animal类
- 写了两个校验方法,分别放到了两个类CheckIfHopper、CheckIfSwimmer里面
- 写了一个接口CheckTraits,让上面的这两个类实现自这个接口
- 在print方法里调用
print(animals, new CheckIfHopper())
或print(animals, new CheckIfSwimmer())
做了这么多铺垫,所以Lambda表达式到底是什么?
首先思考一个问题,根据上面判断跳和游泳的思路,如果想增加一个是否能飞的判断,需要做什么?
- 创建一个CheckIfFlyer类
- 实现CheckTraits接口
- 设置
test()
方法返回值为a.canFly
(还记得这个test方法吗?这个就是我们校验能力的接口里唯一的一个方法名) - 在print方法里调用
print(animals, new CheckIfFlyer())
虽然这样做也不会增加几行代码,也就是多创建一个类的事情。但是Lambda表达式可以做到更简单:
- 在print方法里调用
print(ainmals, a -> a.canFly())
或者print(ainmals, (Animal a) -> {return a.canFly();))
看到了吗?就是这样,并不需要创建一个新的类,也不需要修改test()
方法。
我们来看一下Lambda表达式的结构:
- 左边的
a
是Lambda表达式的唯一参数。 - 中间的箭头
->
用来分隔参数和主体 - 右边的
a.canHop()
是主体,调用唯一方法并返回方法的返回值
上图的结构式Lambda表达式的简洁形式,另外一种更为全面和复杂(并不):
- 当且仅当有且只有一个参数的时候,括号和参数类型才可以被省略
- 主体部分只有一条语句的时候,Lambda表达式不但可以让你省掉花括号
{ }
,还可以让你省略掉return
和后面的分号;
- 主体部分有多条语句构成一个语句块的时候,花括号内分号不可省略
现在我们了解了Lambda的形式,那么Lambda具体是怎么发挥作用的呢?这要从我们的print()
方法说起:
private static void print(List<Animal> animals, CheckTraits checker)
当我们调用print(ainmals, a -> a.canFly()
的时候,我们是将Lambda表达式a -> a.canFly()
作为print方法的第二个参数传过去的。
print方法的第二个参数接收类型为CheckTraits的变量,所以Java会尝试将Lambda表达式映射到CheckTraits接口上:
boolean test(Animal a);
- 因为test方法接收Animal类型的变量,所以Lambda表达式的参数也必须为Animal类型
- 因为test方法的返回值是布尔型,所以Lambda主体也会返回布尔型的值
所以这里其实可以理解为,把简略形式的a -> a.canFly()
映射为了具体形式的(Animal a) -> {return a.canFly();}
这里用到了一种叫延迟执行(Deferred Execution)的概念,所谓延迟执行就是”先定义,再执行“。在调用的时候定义了参数的类型和返回值,等执行到print方法里的时候再去执行主体获得返回值。
关于Lambda还有很多更复杂的应用,比如可以使用Predicate<T>
接口来省掉自己写接口的步骤,感兴趣可以进一步详细学习。
参考资料:OCA: Oracle Certified Associate Java SE 8 Programmer I Study Guide
Java中的Lambda表达式简介及应用的更多相关文章
- Lambda 表达式,Java中应用Lambda 表达式
一.Lambda 表达式 简单来说,编程中提到的 lambda 表达式,通常是在需要一个函数,但是又不想费神去命名一个函数的场合下使用,也就是指匿名函数. 链接:知乎 先举一个普通的 Python 例 ...
- Java中的Lambda表达式
Lambda来源于希腊字母入,发音为 /'læmdə/对高数有所了解的人都知道λ用于声明一个数学逻辑系统,表示根据XX的输入参数,会返回某个Y结果.这正是编程语言中函数(方法)的意思.因此Lambd ...
- java中使用Lambda表达式的5种语法
1,标准写法 思考下述情况: String[] arr = {"program", "creek", "is", "a" ...
- Java 中的 Lambda 表达式
Lambda表达式 Lambda 表达式是 JDK1.8 的一个新特性,又称特殊的匿名内部类,可以取代大部分的匿名内部类,语法更简洁,可以写出更优雅的 Java 代码,可以极大地优化代码结构. Lam ...
- java中使用lambda表达式
使用lambda表达式能够使复杂的编写方式变的简单 lambda表达式的语法 (parameters) -> expression 或 (parameters) ->{ statement ...
- 理解和运用Java中的Lambda
前提 回想一下,JDK8是2014年发布正式版的,到现在为(2020-02-08)止已经过去了5年多.JDK8引入的两个比较强大的新特性是Lambda表达式(下文的Lambda特指JDK提供的Lamb ...
- Java中的lambda匿名函数使用
Java中的lambda匿名函数使用 lambda匿名函数的使用是为了满足某些情况下需要临时定义函数,或者事先定义,需要时才使用.在python里面,lambda表达式的表达方式为:lambda 参数 ...
- 在Android中使用Java 8的lambda表达式
作为一名Java开发者,或许你时常因为缺乏闭包而产生许多的困扰.幸运的是:Java's 8th version introduced lambda functions给我们带来了好消息;然而,这咩有什 ...
- Android中使用lambda表达式
lambda 语法简介 视频为本篇播客知识点讲解,建议采用超清模式观看, 欢迎点击订阅我的优酷 如果刚学Android,不知道怎么写点击事件可以跳转,传送门 要想在Android中使用lambda语法 ...
随机推荐
- 什么是MonoGame?
MonoGame是XNA的一个开源实现.主要用于游戏开发. 官方网站:http://www.monogame.net/ 源码地址:https://github.com/MonoGame/MonoGam ...
- 毕设(五)ListView
ListView 控件可使用四种不同视图显示项目.通过此控件,可将项目组成带有或不带有列标头的列,并显示伴随的图标和文本. 可使用 ListView 控件将称作 ListItem 对象的列表条目组织成 ...
- PRML Chapter2
参考文献:PRML2 参数方法和非参数方法 机器学习上的方法分为参数方法(根据先验知识假定模型服从某种分布,然后利用训练集估计出模型参数,也就弄清楚了整个模型,例如感知器)和非参数方法(基于记忆训练集 ...
- 三种扩展 Office 软件功能的开发模型对比 – Office Add-In Model, VBA 和 VSTO
当 Office 用户需要针对文档自定义新功能时,可以求助于 VBA 或者 VSTO 两种方式.Office 2013 富客户端以后,微软为 Office 平台上的开发者提供了一种新模型 --- Of ...
- [android自动化构建]之centos安装gradle
这是android自动化构建系列之环境配置 这里只记录部分gradle相关的配置 下载并解压 下载地址参考这里:https://services.gradle.org/distributions/,未 ...
- 安装使用Cloudera Impala
安装与使用Cloudera Impala Cloudera Impala提供快速的.交互式的SQL查询方式,直接基于Apache Hadoop存储在HDFS或HBase中的数据进行查询.除了使用与Ap ...
- 妹子问我maven是啥?从相亲说起。。
自从上一篇原创文章: 第一次教妹子安装IDEA 在<java技术之家>公号发表之后,大家的好评如潮,这给了我继续写下去的信心.感谢你们的支持,我会继续努力的. 自从漂亮妹妹加入我们研发团队 ...
- 【maven 】jar包冲突-记一次冲突解决
方法一:根据mvn提示一个一个排除 1.请到pom.xml文件所在的目录(包含父子目录)下分别执行下面的命令排查是什么原因导致fastjson版本不正确: mvn dependency:tree -D ...
- Input标签中Type的类型及详细用法
Input表示Form表单中的一种输入对象,其又随Type类型的不同而分文本输入框,密码输入框,单选/复选框,提交/重置按钮等,下面一一介绍.1,type=text输入类型是text,这是我们见的最多 ...
- flask模板语言 jinja2 以及render_template 深度用法
是时候开始写个前端了,Flask中默认的模板语言是Jinja2 现在我们来一步一步的学习一下 Jinja2 捎带手把 render_template 中留下的疑问解决一下 首先我们要在后端定义几个字符 ...