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

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

上一篇博客中,我们一起学习了行为参数化这个有趣的概念。同时我们也知道了在 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思源笔记

    1083 引用 • 3461 回帖 • 262 关注
  • Java

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

    3169 引用 • 8208 回帖
  • Lambda
    23 引用 • 19 回帖

相关帖子

欢迎来到这里!

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

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

推荐标签 标签

  • FreeMarker

    FreeMarker 是一款好用且功能强大的 Java 模版引擎。

    23 引用 • 20 回帖 • 437 关注
  • 自由行
    2 关注
  • 微服务

    微服务架构是一种架构模式,它提倡将单一应用划分成一组小的服务。服务之间互相协调,互相配合,为用户提供最终价值。每个服务运行在独立的进程中。服务于服务之间才用轻量级的通信机制互相沟通。每个服务都围绕着具体业务构建,能够被独立的部署。

    96 引用 • 155 回帖
  • Shell

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

    122 引用 • 73 回帖 • 1 关注
  • 架构

    我们平时所说的“架构”主要是指软件架构,这是有关软件整体结构与组件的抽象描述,用于指导软件系统各个方面的设计。另外还有“业务架构”、“网络架构”、“硬件架构”等细分领域。

    140 引用 • 441 回帖
  • 正则表达式

    正则表达式(Regular Expression)使用单个字符串来描述、匹配一系列遵循某个句法规则的字符串。

    31 引用 • 94 回帖
  • 安装

    你若安好,便是晴天。

    131 引用 • 1184 回帖
  • Webswing

    Webswing 是一个能将任何 Swing 应用通过纯 HTML5 运行在浏览器中的 Web 服务器,详细介绍请看 将 Java Swing 应用变成 Web 应用

    1 引用 • 15 回帖 • 632 关注
  • iOS

    iOS 是由苹果公司开发的移动操作系统,最早于 2007 年 1 月 9 日的 Macworld 大会上公布这个系统,最初是设计给 iPhone 使用的,后来陆续套用到 iPod touch、iPad 以及 Apple TV 等产品上。iOS 与苹果的 Mac OS X 操作系统一样,属于类 Unix 的商业操作系统。

    84 引用 • 139 回帖
  • 禅道

    禅道是一款国产的开源项目管理软件,她的核心管理思想基于敏捷方法 scrum,内置了产品管理和项目管理,同时又根据国内研发现状补充了测试管理、计划管理、发布管理、文档管理、事务管理等功能,在一个软件中就可以将软件研发中的需求、任务、bug、用例、计划、发布等要素有序的跟踪管理起来,完整地覆盖了项目管理的核心流程。

    6 引用 • 15 回帖 • 186 关注
  • 小薇

    小薇是一个用 Java 写的 QQ 聊天机器人 Web 服务,可以用于社群互动。

    由于 Smart QQ 从 2019 年 1 月 1 日起停止服务,所以该项目也已经停止维护了!

    34 引用 • 467 回帖 • 711 关注
  • flomo

    flomo 是新一代 「卡片笔记」 ,专注在碎片化时代,促进你的记录,帮你积累更多知识资产。

    4 引用 • 91 回帖
  • 微信

    腾讯公司 2011 年 1 月 21 日推出的一款手机通讯软件。用户可以通过摇一摇、搜索号码、扫描二维码等添加好友和关注公众平台,同时可以将自己看到的精彩内容分享到微信朋友圈。

    130 引用 • 793 回帖 • 1 关注
  • Scala

    Scala 是一门多范式的编程语言,集成面向对象编程和函数式编程的各种特性。

    13 引用 • 11 回帖 • 111 关注
  • 新人

    让我们欢迎这对新人。哦,不好意思说错了,让我们欢迎这位新人!
    新手上路,请谨慎驾驶!

    51 引用 • 226 回帖
  • H2

    H2 是一个开源的嵌入式数据库引擎,采用 Java 语言编写,不受平台的限制,同时 H2 提供了一个十分方便的 web 控制台用于操作和管理数据库内容。H2 还提供兼容模式,可以兼容一些主流的数据库,因此采用 H2 作为开发期的数据库非常方便。

    11 引用 • 54 回帖 • 649 关注
  • SVN

    SVN 是 Subversion 的简称,是一个开放源代码的版本控制系统,相较于 RCS、CVS,它采用了分支管理系统,它的设计目标就是取代 CVS。

    29 引用 • 98 回帖 • 689 关注
  • ZeroNet

    ZeroNet 是一个基于比特币加密技术和 BT 网络技术的去中心化的、开放开源的网络和交流系统。

    1 引用 • 21 回帖 • 607 关注
  • 阿里云

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

    89 引用 • 345 回帖
  • GitLab

    GitLab 是利用 Ruby 一个开源的版本管理系统,实现一个自托管的 Git 项目仓库,可通过 Web 界面操作公开或私有项目。

    46 引用 • 72 回帖
  • golang

    Go 语言是 Google 推出的一种全新的编程语言,可以在不损失应用程序性能的情况下降低代码的复杂性。谷歌首席软件工程师罗布派克(Rob Pike)说:我们之所以开发 Go,是因为过去 10 多年间软件开发的难度令人沮丧。Go 是谷歌 2009 发布的第二款编程语言。

    493 引用 • 1385 回帖 • 341 关注
  • BND

    BND(Baidu Netdisk Downloader)是一款图形界面的百度网盘不限速下载器,支持 Windows、Linux 和 Mac,详细介绍请看这里

    107 引用 • 1281 回帖 • 31 关注
  • JavaScript

    JavaScript 一种动态类型、弱类型、基于原型的直译式脚本语言,内置支持类型。它的解释器被称为 JavaScript 引擎,为浏览器的一部分,广泛用于客户端的脚本语言,最早是在 HTML 网页上使用,用来给 HTML 网页增加动态功能。

    713 引用 • 1174 回帖 • 119 关注
  • 30Seconds

    📙 前端知识精选集,包含 HTML、CSS、JavaScript、React、Node、安全等方面,每天仅需 30 秒。

    • 精选常见面试题,帮助您准备下一次面试
    • 精选常见交互,帮助您拥有简洁酷炫的站点
    • 精选有用的 React 片段,帮助你获取最佳实践
    • 精选常见代码集,帮助您提高打码效率
    • 整理前端界的最新资讯,邀您一同探索新世界
    488 引用 • 383 回帖 • 3 关注
  • 星云链

    星云链是一个开源公链,业内简单的将其称为区块链上的谷歌。其实它不仅仅是区块链搜索引擎,一个公链的所有功能,它基本都有,比如你可以用它来开发部署你的去中心化的 APP,你可以在上面编写智能合约,发送交易等等。3 分钟快速接入星云链 (NAS) 测试网

    3 引用 • 16 回帖 • 1 关注
  • SendCloud

    SendCloud 由搜狐武汉研发中心孵化的项目,是致力于为开发者提供高质量的触发邮件服务的云端邮件发送平台,为开发者提供便利的 API 接口来调用服务,让邮件准确迅速到达用户收件箱并获得强大的追踪数据。

    2 引用 • 8 回帖 • 445 关注
  • 创造

    你创造的作品可能会帮助到很多人,如果是开源项目的话就更赞了!

    175 引用 • 992 回帖 • 1 关注