读书笔记——《Java 8 实战》系列之 Lambda 表达式 (一)

本贴最后更新于 2746 天前,其中的信息可能已经事过境迁

上一篇博客中,我们一起学习了行为参数化这个有趣的概念。同时我们也知道了在 Java 8 之前,匿名类可以用来减少那些只使用一次的实体类的啰嗦代码。

而 Java 8 中,Lambda 表达式的出现能够让我们以一种更加简洁的方式去表示一个行为或传递代码。

简单来说,Lambda 表达式是一种匿名机制,它是一种没有声明名称的方法,和匿名类一样,它也可以作为参数被传递给方法。

还记得上一篇博客中我们给出的 Lambda 表达式的例子么?

不使用匿名类时:

//声明一个实现了StudentPredicate接口的实体类 class StudentHeightPredicate implements StudentPredicate{ @Override public boolean test(Student s) { if(s.getHeight()>=180) return true; return false; } } public static void main(String[] args) { //返回身高超过180cm的学生 List<Student> filteredStudents = studentFilter(students, new StudentHeightPredicate()); }

使用匿名类的话:

public static void main(String[] args) { //返回身高超过180cm的学生 List<Student> filteredStudents = studentFilter(students, new StudentPredicate() { @Override public boolean test(Student s) { if(s.getHeight()>=180) return true; return false; } }); }

使用 Lambda 表达式的话:

public static void main(String[] args){ List<Student> filteredStudents2 = studentFilter(students, (Student s) -> { if(s.getHeight()>=180) return true; return false; } ); }

与上一段使用了匿名类机制的代码相比,使用 Lambda 表达式更像是将 test 方法的主体传入到了 studentFilter()方法当中。

那么接下来我们就一起来学习一下 Lambda 表达式的写法规则。

Lambda 表达式包含以下三个部分:

  • 参数列表——在本例中就是 test 方法中的参数 Student s,被圆括号包围着
  • 箭头——箭头 -> 用来将参数列表与 Lambda 表达式的主题分隔开
  • Lambda 主体——也就是使用匿名类代码中 test 方法中的实现代码

在大家都了解了 Lambda 表达式的语法规则后,我们给出一些 Lambda 表达式的例子,大家可以从中学习到如何简化使用 Lambda 表达式

例子 1:

(String s) -> s.length()

*tips: 参数为 String 类型的 s ,方法的主体返回了一个 int 类型的值 s.length(),当方法主体只返回一个值时,主体无需被花括号包围,同时 return 可以被省略

例子 2:

(Student s) -> s.getHeight()>=180

*tips 参数为 Student 类型的 s, 方法的主体返回了一个 Boolean 类型的判断 s.getHeight()>=180, 仔细观察这个表达式其实就是我们上面给 studentFilter 例子的简化版,
大家可以学习一下这种简便的写法

例子 3:

(int x, int y) -> { System.out.println("Result:"); System.out.println(x+y); }

*tips 参数为 2 个 int 类型的值,方法的主体是两句打印输出,同时方法的主体中没有返回值,需要注意的是,当使用花括号包围起方法主体时不能忘记每句语句结尾的分号

例子 4:

() -> 42

*tips 没有参数,直接返回一个 int 类型的值 42

例子 5:

s -> s.getHeight()>=180

*tips 例子 2 的简化版,当 Java 的编译器嗯能够根据上下文去判断参数 s 的类型时,可以省略参数类型,同时当参数只有一个的时候,可以省略包围参数列表的圆括号

那么看了这么多 Lambda 表达式的例子,也该考验一下大家是否真的完全掌握了 Lambda 表达式的写法,判断下列的 Lambda 表达式中哪些是合法的哪些是非法的。

测验:

  1. () -> { }
  2. String s -> "hello world"
  3. (s,s2) -> {return "hello world";}
  4. s -> { "hello world" }
  5. (String s) -> { return "hello world"}

解析

  1. 第一个表达式合法,参数为空,返回值为空
  2. 第二个表达式非法,在参数数量只有一个但是没有省略参数类型时,圆括号不能省略
  3. 第三个表达式合法,Java 编译器推断出 s,s2 的参数类型,则可省略,在方法主体内可以显式地使用 return 关键字
  4. 第四个表达式非法,在方法主体花括号内返回值必须由 return 关键字显式返回,若没有花括号则可以省略 return 关键字
  5. 第五个表达式非法,在方法主体花括号内 return 语句之后缺少分号

怎么样,大家有没有被这些 Lambda 表达式测验题难住呢?

下表给出了一些 Lambda 表达式的例子和使用案例

使用案例Lambda示例
布尔表达式(List<String> list) -> list.isEmpty()
创建对象( ) -> new String()
消费一个对象( String s ) -> { System.out.println(s);}
从某个对象中抽取属性( Student s ) -> s.getHeight()
组合两个值( int a, int b ) -> a*b
比较两个对象( Student s1, Student s2 ) -> s1.getHeight().compareTo(s2.getHeight())

学习了这么多 Lambda 表达式相关的知识,同学们肯定会问,我们究竟在哪里能使用到它呢?

Lambda 表达式的一个重要使用场景就是在函数式接口上,有些同学可能对这个函数式接口没有概念,别担心,接下来我们首先了解一下这个函数式接口究竟是什么

还记得上一篇博客中,我们写的 Predicate这个接口么?它就是一个典型的函数式接口,因为在接口中我们仅定义了一个抽象方法:

interface Predicate<T>{ //这里使用泛型来传入对象 boolean test(T t); }

简单来说,**函数式接口就是只定义了一个抽象方法的接口。**回想一下,在平日里我们经常接触到的函数接口有哪些?

public interface Comparator<T> { int compareTo(T o1, T o2); }
public interface Runnable{ void run(); }
public interface Callable<V>{ V call(); }

为了检查你的理解程度,下面的测验能够帮助你测试是否掌握了函数式接口的概念:

下列哪些接口是函数式接口?

public interface Adder{ int add(int a, int b); } public interface SmartAdder extends Adder{ int add(double a, double b); } public interface Nothing{ }

解析:

只有 Adder 接口是函数式接口,SmartAdder 从 Adder 接口中继承了一个 add 方法,因此不是函数式接口,Nothing 接口也不是函数式接口,因为它并没有声明方法。

在 Java 8 中,Lambda 表达式允许你直接以内联的形式为函数式接口的抽象方法提供实现,并把整个表达式作为函数式接口的实例。尽管我们匿名类也可以完成同样的事,但是却比较笨重,因为我们不得不先提供一个具体实现,再直接将它内联实例化。有的同学看到这里可能会有点晕,没关系,下面我就给大家提供一些例子来理解这段话。

//使用匿名类声明一个Runnable接口的实例 Runnable r1 = new Runnable(){ public void run(){ System.out.println("hello world 1"); } }; //使用Lambda表达式声明一个Runnable接口的实例 Runnable r2 = () -> System.out.println("hello world 2"); //打印hello world 1 new Thread(r1).start(); //打印hello world 2 new Thread(r2).start(); //直接将Lambda表达式当作参数,打印 hello world 3 new Thread(() -> System.out.println("hello world 3")).start();

关注新的 Java API 的同学会发现,函数式接口在声明时会伴随着 @FunctionalInterface 的标注。当我们使用了这个标注,但是接口中确声明了超过一个的抽象方法时,编译器就会报错。当然这个标注并不是必须的,我么可以定义只有一个抽象方法的函数式接口但是却不加这个标注。

接下来,我给大家准备了一个常见的编程模式,让大家在实战中增强对行为参数化Lambda 表达式的运用。

不知道同学们有没有注意到,在编程中我们经常会遇到以下的一种情形:

//1.完成一些固定的前期工作 doSomeStartingWork(); //2.完成真正的我们需要做的事,这些事往往是不固定的,可变的 doActualWork(); //3.完成一些固定的结尾工作 doSomeEndingWork();

给大家举个简单的例子吧:

public static void main(String[] args) throws IOException { // TODO Auto-generated method stub //完成一些固定的前期工作,初始化一个BufferReader与需要打开的资源 BufferedReader br = new BufferedReader(new FileReader("data.txt")); //完成我们真正需要做的工作,这里仅仅是打印了一行内容,这些工作是可能发生改变的,比如变成打印两行内容 System.out.println(br.readLine()); //完成一些固定的结尾工作,关闭资源 br.close(); }

在上面的例子中,这个打开资源与关闭资源是固定不变的,可变的仅仅是我们需要对资源做怎样的处理。再比如,我们经常需要测试一段代码的执行时间,我们会怎么做呢?记录开始时间,记录结束时间,两者相减,可变的仅仅是我们需要执行的代码。像这种开始与结尾总是固定,只有中间需要处理的重要代码可变的模式,我们通常称之为环绕执行(Execute Around)模式。

看到这种模式的特点之后,大家有没有觉得很眼熟。这不正式行为参数化最擅长干的事情么,将重要的代码变成参数传递给方法,以抽象应对改变

那接下来,博主就一步步带着大家完成一个测试代码执行时间的例子。

  • 首先我们定义一个函数式接口
//在这个接口中,我们只声明了一个没有任何返回值,不需要任何参数的抽象方法,这个方法的作用就是为了执行我们需要知道执行时间的代码 @FunctionalInterface interface Executor{ void execute(); }
  • 接下来,我们定义我们的主方法
public static long getExecutionTime(Executor e) { //记录起始时间 long startTime = System.currentTimeMillis(); //等待被执行代码执行 e.execute(); //记录结束时间 long endTime = System.currentTimeMillis(); //返回总共消耗的时间,并以秒为单位 return (endTime - startTime)/ 1000; }
  • 然后我们就可以获得我们需要执行代码的执行时间了
public static void main(String[] args) { //在getExecutionTime()方法中,使用Lambda表达式作为它的参数,在这个例子中我们什么也没做,只是让主线程Sleep了5秒 long executionTime = getExecutionTime(() -> {try { Thread.sleep(5000); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); }}); System.out.println("代码执行时间为: " +executionTime +"s"); }

学了这么多关于 Lambda 表达式的知识,大家也需要消化一下了,其它关于 Lambda 表达式的知识将在下一篇博客中与大家分享。

  • B3log

    B3log 是一个开源组织,名字来源于“Bulletin Board Blog”缩写,目标是将独立博客与论坛结合,形成一种新的网络社区体验,详细请看 B3log 构思。目前 B3log 已经开源了多款产品:SymSoloVditor思源笔记

    1063 引用 • 3455 回帖 • 161 关注
  • Java

    Java 是一种可以撰写跨平台应用软件的面向对象的程序设计语言,是由 Sun Microsystems 公司于 1995 年 5 月推出的。Java 技术具有卓越的通用性、高效性、平台移植性和安全性。

    3200 引用 • 8216 回帖
  • Lambda
    24 引用 • 19 回帖

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...

推荐标签 标签

  • Q&A

    提问之前请先看《提问的智慧》,好的问题比好的答案更有价值。

    9713 引用 • 44202 回帖 • 90 关注
  • 运维

    互联网运维工作,以服务为中心,以稳定、安全、高效为三个基本点,确保公司的互联网业务能够 7×24 小时为用户提供高质量的服务。

    150 引用 • 257 回帖 • 1 关注
  • Sym

    Sym 是一款用 Java 实现的现代化社区(论坛/BBS/社交网络/博客)系统平台。

    下一代的社区系统,为未来而构建

    524 引用 • 4601 回帖 • 708 关注
  • TGIF

    Thank God It's Friday! 感谢老天,总算到星期五啦!

    290 引用 • 4494 回帖 • 653 关注
  • 域名

    域名(Domain Name),简称域名、网域,是由一串用点分隔的名字组成的 Internet 上某一台计算机或计算机组的名称,用于在数据传输时标识计算机的电子方位(有时也指地理位置)。

    43 引用 • 208 回帖
  • abitmean

    有点意思就行了

    37 关注
  • BookxNote

    BookxNote 是一款全新的电子书学习工具,助力您的学习与思考,让您的大脑更高效的记忆。

    笔记整理交给我,一心只读圣贤书。

    1 引用 • 1 回帖
  • Pipe

    Pipe 是一款小而美的开源博客平台。Pipe 有着非常活跃的社区,可将文章作为帖子推送到社区,来自社区的回帖将作为博客评论进行联动(具体细节请浏览 B3log 构思 - 分布式社区网络)。

    这是一种全新的网络社区体验,让热爱记录和分享的你不再感到孤单!

    133 引用 • 1124 回帖 • 111 关注
  • 阿里云

    阿里云是阿里巴巴集团旗下公司,是全球领先的云计算及人工智能科技公司。提供云服务器、云数据库、云安全等云计算服务,以及大数据、人工智能服务、精准定制基于场景的行业解决方案。

    84 引用 • 324 回帖
  • IDEA

    IDEA 全称 IntelliJ IDEA,是一款 Java 语言开发的集成环境,在业界被公认为最好的 Java 开发工具之一。IDEA 是 JetBrains 公司的产品,这家公司总部位于捷克共和国的首都布拉格,开发人员以严谨著称的东欧程序员为主。

    181 引用 • 400 回帖
  • CloudFoundry

    Cloud Foundry 是 VMware 推出的业界第一个开源 PaaS 云平台,它支持多种框架、语言、运行时环境、云平台及应用服务,使开发人员能够在几秒钟内进行应用程序的部署和扩展,无需担心任何基础架构的问题。

    5 引用 • 18 回帖 • 181 关注
  • 单点登录

    单点登录(Single Sign On)是目前比较流行的企业业务整合的解决方案之一。SSO 的定义是在多个应用系统中,用户只需要登录一次就可以访问所有相互信任的应用系统。

    9 引用 • 25 回帖 • 1 关注
  • 友情链接

    确认过眼神后的灵魂连接,站在链在!

    24 引用 • 373 回帖
  • C++

    C++ 是在 C 语言的基础上开发的一种通用编程语言,应用广泛。C++ 支持多种编程范式,面向对象编程、泛型编程和过程化编程。

    107 引用 • 153 回帖
  • 招聘

    哪里都缺人,哪里都不缺人。

    188 引用 • 1057 回帖
  • Laravel

    Laravel 是一套简洁、优雅的 PHP Web 开发框架。它采用 MVC 设计,是一款崇尚开发效率的全栈框架。

    20 引用 • 23 回帖 • 741 关注
  • DevOps

    DevOps(Development 和 Operations 的组合词)是一组过程、方法与系统的统称,用于促进开发(应用程序/软件工程)、技术运营和质量保障(QA)部门之间的沟通、协作与整合。

    58 引用 • 25 回帖 • 2 关注
  • PWA

    PWA(Progressive Web App)是 Google 在 2015 年提出、2016 年 6 月开始推广的项目。它结合了一系列现代 Web 技术,在网页应用中实现和原生应用相近的用户体验。

    14 引用 • 69 回帖 • 177 关注
  • 代码片段

    代码片段分为 CSS 与 JS 两种代码,添加在 [设置 - 外观 - 代码片段] 中,这些代码会在思源笔记加载时自动执行,用于改善笔记的样式或功能。

    用户在该标签下分享代码片段时需在帖子标题前添加 [css] [js] 用于区分代码片段类型。

    162 引用 • 1082 回帖
  • JWT

    JWT(JSON Web Token)是一种用于双方之间传递信息的简洁的、安全的表述性声明规范。JWT 作为一个开放的标准(RFC 7519),定义了一种简洁的,自包含的方法用于通信双方之间以 JSON 的形式安全的传递信息。

    20 引用 • 15 回帖 • 20 关注
  • 前端

    前端技术一般分为前端设计和前端开发,前端设计可以理解为网站的视觉设计,前端开发则是网站的前台代码实现,包括 HTML、CSS 以及 JavaScript 等。

    245 引用 • 1338 回帖 • 1 关注
  • OneNote
    1 引用 • 3 回帖
  • Electron

    Electron 基于 Chromium 和 Node.js,让你可以使用 HTML、CSS 和 JavaScript 构建应用。它是一个由 GitHub 及众多贡献者组成的活跃社区共同维护的开源项目,兼容 Mac、Windows 和 Linux,它构建的应用可在这三个操作系统上面运行。

    15 引用 • 136 回帖 • 4 关注
  • Shell

    Shell 脚本与 Windows/Dos 下的批处理相似,也就是用各类命令预先放入到一个文件中,方便一次性执行的一个程序文件,主要是方便管理员进行设置或者管理用的。但是它比 Windows 下的批处理更强大,比用其他编程程序编辑的程序效率更高,因为它使用了 Linux/Unix 下的命令。

    125 引用 • 74 回帖
  • 倾城之链
    23 引用 • 66 回帖 • 167 关注
  • Jenkins

    Jenkins 是一套开源的持续集成工具。它提供了非常丰富的插件,让构建、部署、自动化集成项目变得简单易用。

    54 引用 • 37 回帖
  • Redis

    Redis 是一个开源的使用 ANSI C 语言编写、支持网络、可基于内存亦可持久化的日志型、Key-Value 数据库,并提供多种语言的 API。从 2010 年 3 月 15 日起,Redis 的开发工作由 VMware 主持。从 2013 年 5 月开始,Redis 的开发由 Pivotal 赞助。

    286 引用 • 248 回帖