Java 8 中的 Lambda 表达式
Lambda 表达式是 Java 8 最受欢迎的功能。人们将函数式编程的概念引入了 Java 这门完全面向对象的命令式编程语言。
关于函数式编程是如何运作的,这个话题超出了本文的范围,不过我们会提炼出它一个明显有别于我们所经常使用的 OOP (面向对象编程)的功能来加以讨论。
在本文中, 我们将了解到 lambda 表达式具体是什么东西,还有就是它们是如何将自己融入整个 Java 生态系统的。我们也会对没有使用 lambda 表达式的代码以及后面使用 lambda 进行重构的示例代码进行一下观察和比较。
了解 Lambda 表达式
Lambda 表达式是一块我们可以将其传入并执行的代码。对于作为 Java 程序员的我们而言,并不会怎么习惯将一块代码传入一个函数这样的方式。我们的习惯是将定义的代码封装到方法体里面,然后通过对象引用来加以执行,如下所示:
public class LambdaDemo { public void printSomething(String something) { System.out.println(something); } public static void main(String[] args) { LambdaDemo demo = new LambdaDemo(); String something = "I am learning Lambda"; demo.printSomething(something); } }
这是经典 OOP 开发范式的风格,将方法实现对调用者隐藏。调用者只是简单地向方法传入一个变量,然后方法拿这个变量会执行一些操作,并返回另外一个变量值,或者如我们的示例所示,会产生一些副作用效果。
现在我们要来看看一种使用了行为传递方式,而不是变量传递的等效实现。为此,我们要创建一个函数式的接口,里面定义的是对行为,而不是对方法的抽象。一个函数式接口是一种只有一个方法的接口:
public class LambdaDemo { interface Printer { void print(String val); } public void printSomething(String something, Printer printer) { printer.print(something); } }
在上面的代码实现中, Printer 接口负责所有的打印操作。printSomething 方法不再对行为进行定义,而是执行由 Printer 定义的行为:
public static void main(String[] args) { LambdaDemo demo = new LambdaDemo(); String something = "I am using a Functional interface"; Printer printer = new Printer() { @Override public void print(String val) { System.out.println(val); } }; demo.printSomething(something, printer); }
读者中比较有观察能力的可能已经注意到,我并没有在这里做什么新的事情。的确是这样的,因为我还没有应用到 lambda 表达式。我们只是简单地创建了一个 Printer 接口的具体实现,并将它传入了 printSomething 方法。
上面的示例旨在给我们带来一个将 Lambda 表达式引入到 Java 中的关键目标:
Lambda 表达式原被用于定义一个函数式接口的内联实现。
在我们使用 lambda 表达式对上面的示例进行重构之前,先来学习一下必要的语法知识:
(param1,param2,param3...,paramN) - > {//代码块;}
一个 lambda 表达式的组成,是一个我们通常会定义在方法声明中的,以括弧封闭起来并以逗号分隔的参数列表,后面跟上一个箭头标记指向要执行的代码。现在,让我们来使用 lambda 对上面的代码进行重构:
public static void main(String[] args) { LambdaDemo demo = new LambdaDemo(); String something = "I am learning Lambda"; /**/ Printer printer = (String toPrint)->{System.out.println(toPrint);}; /**/ demo.printSomething(something, printer); }
看上去非常紧凑且美观。因为函数式接口只声明了一个方法,所以在 lambda 的第一部分中传入的参数被自动地映射到了方法的参数列表上,而箭头右边的代码则被当做是方法的具体实现了。
为什么要使用 Lambda 表达式
如同前面的示例, lambda 表达式能让我们拥有更加紧凑的代码,更加易于阅读和跟踪。这个在性能和多核处理方法还有其它的一些好处,不过它们得在你了解了 Streams API 以后才有用,而这个超出了本文的范围。
通过比较使用和没使用 lambda 的 main 方式实现,当它一下子把代码变得简短的时候,我们切实地看到了 lambda 表达式的能力:
public static void main(String[] args) { LambdaDemo demo = new LambdaDemo(); String something = "I am learning Lambda"; /**/ Printer printer = (String toPrint)->{System.out.println(toPrint);}; /**/ demo.printSomething(something, printer); }
我们还可以让代码比这里所展示的更简洁。这样的事情发生时,你甚至无需指定箭头左边参数的类型,而其类型会由编译器根据接口方法的形参推断出来。
Printer printer = (toPrint)->{System.out.println(toPrint);};
我们还可以做得更好。lambda 的另外一个特性就是: 如果只有一个参数, 就可以将括弧完全消除掉。同样的,如果在箭头右边只有一条语句,也可以将大括号去掉:
Printer printer = toPrint -> System.out.println(toPrint);
现在的代码看起来真正变得可爱起来,不过我们才刚刚开始而已。如果我们的接口方法并不要任何参数,那就可以将生命用一对空的括弧替换掉:
() -> System.out.println("anything");
如果我们只是内联一个 lambda 进去,而不去首先创建一个对象然后将其传入到 saySomething 方法,会如何呢:
public static void main(String[] args) { LambdaDemo demo = new LambdaDemo(); String something="I am Lambda"; /**/ demo.printSomething(something, toPrint -> System.out.println(toPrint)); }
现在我们才是真的在谈论函数式编程了。我们的 main 函数体从一开始的 9 行代码减少到了 3 行。这样紧凑的代码使得 lambda 表达式对于 Java 程序员非常有吸引力。
Java 8 中的 Lambda 表达式的更多相关文章
- java函数式编程之lambda表达式
作为比较老牌的面向对象的编程语言java,在对函数式编程的支持上一直不温不火. 认为面向对象式编程就应该纯粹的面向对象,于是经常看到这样的写法:如果你想写一个方法,那么就必须把它放到一个类里面,然后n ...
- Lambda 表达式,Java中应用Lambda 表达式
一.Lambda 表达式 简单来说,编程中提到的 lambda 表达式,通常是在需要一个函数,但是又不想费神去命名一个函数的场合下使用,也就是指匿名函数. 链接:知乎 先举一个普通的 Python 例 ...
- Java中的Lambda表达式简介及应用
在接触Lambda表达式.了解其作用之前,首先来看一下,不用Lambda的时候我们是怎么来做事情的. 我们的需求是,创建一个动物(Animal)的列表,里面有动物的物种名,以及这种动物是否会跳,是否会 ...
- C#中的Lambda表达式和表达式树
在C# 2.0中,通过方法组转换和匿名方法,使委托的实现得到了极大的简化.但是,匿名方法仍然有些臃肿,而且当代码中充满了匿名方法的时候,可读性可能就会受到影响.C# 3.0中出现的Lambda表达式在 ...
- [Java 8] (5) 使用Lambda表达式进行设计
使用Lambda表达式进行设计 在前面的几篇文章中,我们已经见识到了Lambda表达式是怎样让代码变的更加紧凑和简洁的. 这一篇文章主要会介绍Lambda表达式怎样改变程序的设计.怎样让程序变的更加轻 ...
- Java 高级特性: Lambda 表达式
本文源代码上传到了码云,请点击 LambdaExpression 获取.Lambda 表达式是 java 8 的新特性,本文讲解了 lambda 表达式的所有知识.内容涉及到 lambda 表达式是什 ...
- Java函数式编程和lambda表达式
为什么要使用函数式编程 函数式编程更多时候是一种编程的思维方式,是种方法论.函数式与命令式编程的区别主要在于:函数式编程是告诉代码你要做什么,而命令式编程则是告诉代码要怎么做.说白了,函数式编程是基于 ...
- Android中使用Lambda表达式开发
参考文章:ImportNew 要在Android开发中使用lambda表达式,首先需要在 Module 的build.gradle中加入: compileOptions { targetCompati ...
- Java核心技术-接口、lambda表达式与内部类
本章将主要介绍: 接口技术:主要用来描述类具有什么功能,而并不给出每个功能的具体实现.一个类可以实现一个或多个接口. lambda表达式:这是一种表示可以在将来的某个时间点执行的代码块的简洁方法. 内 ...
随机推荐
- python序列化
一. 序列化 1 定义: 在我们存储数据或者⽹网络传输数据的时候. 需要对我们的对象进⾏行行处理理. 把对象处理理成 ⽅方便便存储和传输的数据格式. 这个过程叫序列列化. 不同的序列列化, 结果也不同 ...
- Python 19 Django 详解
本节概要 Django详解 前言 有一部分原因是,确实djando的课程有点多:并且,最近又在研究利用python做数据分析时间上耽误了.所以楼主讲所有的课程全部重新观看了一遍,再来撰写博客,其实说起 ...
- mysql 案例 ~ 表空间迁移数据与数据导入
一 简介:mysql5.6+的表空间传输二 目的:复制数据到另一个表三 步骤 1 create table b like a ->创建一个空表 2 alter table b disc ...
- 20165234 《Java程序设计》第八周学习总结
第八周学习总结 教材内容学习 第十二章 Java 多线程机制 进程与线程 进程是程序的一次动态执行过程,对应了从代码加载.执行至执行完毕的一个完整过程. 线程不是进程,是比进程更小的执行单位. 一个进 ...
- 【转载】NeurIPS 2018 | 腾讯AI Lab详解3大热点:模型压缩、机器学习及最优化算法
原文:NeurIPS 2018 | 腾讯AI Lab详解3大热点:模型压缩.机器学习及最优化算法 导读 AI领域顶会NeurIPS正在加拿大蒙特利尔举办.本文针对实验室关注的几个研究热点,模型压缩.自 ...
- P2805 [NOI2009]植物大战僵尸
题目地址:P2805 [NOI2009]植物大战僵尸 最大权闭合子图 若有向图 \(G\) 的子图 \(V\) 满足: \(V\) 中顶点的所有出边均指向 \(V\) 内部的顶点,则称 \(V\) 是 ...
- SQL Server - GO
GO GO是批处理的标志,是一条或多条SQL语句的集合,SQL Server将批处理语句编译成一个可执行单元,此单元称为执行计划. GO语句把程序分成一个个代码块,即使一个代码块执行错误,它后面的代码 ...
- SpringBoot的Session并发控制
⒈是什么? 即控制业务系统中一个用户只能有一个Session ⒉解决方案 1.当这个用户在其它地方登录的时候,把之前的Session失效掉. package cn.coreqi.security.co ...
- 【转】采用dlopen、dlsym、dlclose加载动态链接库
1.前言 为了使程序方便扩展,具备通用性,可以采用插件形式.采用异步事件驱动模型,保证主控制逻辑不变,将各个业务以动态链接库的形式加载进来,这就是所谓的插件.linux提供了加载和处理动态链接库的系统 ...
- Docker 导出 & 导入
Docker 容器因为它的快速部署被深受喜爱.本文记录 Docker 容器的导出与导入,分别用到 Docker 的 export 和 import 命令. 1.查看正在运行的容器: [root@loc ...