Lambda表达式Java教程

在本文中,我们提供了全面的Lambda Expressions Java教程。

1. Lambda Expressions Java教程–简介

Lambda表达式被认为是Java 8中引入的最好的功能之一。Lambda表达式被认为是Java进入函数式编程世界的第一步 。 可以将其视为无需类即可创建的函数。 它也可以像参数一样传递,并且可以在需要时和根据需要执行。 Java Lambda表达式是匿名函数的简洁表示,可以将其传递。 具有单个功能的匿名类通过额外的语法呈现出笨拙的外观。 这些表述旨在消除这种混乱。

如前所述,Java Lambda表达式是无名函数 ,可以作为常量值进行传递。 这意味着它们可以存在于可能存在任何其他常数值的任何位置,但是通常作为参数写入某些其他函数。 考虑一个典型的例子,我们可以将比较函数传递给泛型排序函数,而不是麻烦地定义一个整个过程(并引起词法不连续和名称空间污染)来描述这种比较,我们只需传递一个lambda表达式描述了比较。 让我们看一下Lambda表达式的一些属性

  • 匿名:它仍然可以称为匿名,因为它没有明确的名称。
  • 简洁:正如前面提到的匿名类的情况,与匿名类相比,我们用Lambdas编写的代码要少得多。
  • 函数:Lambda更像是函数而不是方法。 这是因为方法属于类,而Lambda不属于。 但是就像方法一样,Lambda接受参数列表,具有主体并且还可以引发异常。
  • 可以传递:Lambda可以传递给其他函数,就像普通参数一样。

为了消除由于我们上面提到的观点而引起的任何误解,lambda不会添加引入之前的更多功能。 它只是改善了我们编写代码的方式,并减少了很多样板代码。 该样板代码甚至与我们用来通过基础操作系统的多核性质进行代码识别的系统级编程有关。 让我们看一下这种简单的语法糖如何使我们的工作在并行性,代码简洁性和紧凑性方面更容易。

2.编写Lambda表达式

在本节中,我们将看到Java Lambda表达式如何减少执行一些简单操作所需编写的代码行。 例如,我们将比较代码行数以构成一个Comparator函数。 为了进行比较,我们将在此处创建一个简单的POJO类,即Student类,其中包含Student ID(作为Long和name作为String参数:

学生.java

 public class Student {   private Long id; private String name; // standard setters and getters  } 

比较我们在应用程序中定义的POJO对象是一种非常通用的编程实践。 如果我们要比较两个Student类对象,则可以使Comparator像这样:

匿名类的比较器

 Comparator<Student> byId = new Comparator<Student>() { @Override public int compare(Student s1, Student s2) { return s1.getId().compareTo(s2.getId()); }  }; 

这是一个作为Anonymous类的简单Comparator实现,但是当使用Lambda完成时,我们会发现相同的实现非常精确和干净。 让我们在这里看到使用Lambda表达式完成的相同任务:

pom.xml

 Comparator<Student> byId = (s1, s2) -> s1.getId().compareTo(s2.getId()); 

Lambda表达式上方也可以称为块Lambda表达式,因为它由>符号右侧的单个代码块组成。 它变得更加简洁小巧,这听起来很神奇,请参见以下代码片段:

Lambda的简洁实现

 Comparator<Student> byId = Comparator.comparing(Student::getId); 

这是建立比较器的好方法,而且也很简单。 对于上面我们进行的Block Lambda表达式,让我们将其分为几部分以更好地理解:

Lambda表达式Java-Lambda表达式
Lambda表达
  • Lambda Expression以在此情况下传递给函数Comparator的参数列表开头
  • 箭头符号将Lambda Expression参数与Lambda主体分开
  • 主体清楚地将两个学生对象及其id进行比较,该表达式定义了Lambda返回值

需要注意的是,编译后的代码(即匿名类版本和Lambda表达式版本的字节码)将完全相同,因为Lambda表达式仅仅是使代码清晰的语法。 尽管使用Lambda表达式有时可能会使代码的可读性降低。

3. Lambda表达式与匿名类

我们使用Lambda表达式编写的代码也可以使用Anonymous类编写,其实现方式与Lambda Expressions完全相同。 区别在于Lambda代码的简洁性。

作为比较示例,让我们构造一个类和一个将Runnable作为输入的方法:

可运行类

 public class RunnableInstance { public static void doSomething(Runnable runnable){ runnable.run(); }  } 

当我们使用Anonymous类制作Runnable时,其外观如下所示:

可通过匿名类运行

 Runnable runnable = new Runnable() { @Override public void run() { System.out.print( "Anonymous class implementation." ); }  };  doSomething(runnable); 

让我们尝试将上面的代码转换为Lambda表达式,看看如何得到干净的东西:

可与Lambda一起运行

 Runnable runnable = () -> System.out.print( "Lambda Expression." );  doSomething(runnable); 

如果我们不想多次使用可运行的实现,我们甚至可以避免进行引用:

简洁的Lambda Runnable

 doSomething(() -> System.out.print( "Lambda Expression." )); 

4.使用Lambda表达式进行并行编程

每当我们谈论线程时,我们大多数人都会退后一步,考虑是否真的需要在我们的应用程序中实现线程以支持并行性,因为并行性本质上微不足道且难以管理。 当我们有一个项目集合时,我们实现了一个lambda,如:

并行编程

 collection.map { // my lambda } 

在这里,集合本身能够与提供的Lambda实现并行性,而不必自己实现线程。 这意味着,在多核环境中,Lambda可以在集合上进行流式传输时利用多个核。 就像我们考虑一个简单的例子一样:

Lambda与并行流

 List<String> names = students.stream() .map(s -> s.getName().toUpperCase()) .collect(Collectors.toList()); 

map函数可以在多核环境中并行运行,以一次处理多个对象,而无需我们做任何事情。 为此,仅需要执行此程序的操作系统必须是多核。 一旦满足此条件,我们可以确保可以在给定语句中并行执行的任何操作都将自动完成。

5.收藏和流

Collections框架是Java中最常用的Framework API之一。 集合允许我们将相似的对象收集到可以针对特定目的进行优化的数据结构中。 前面的所有示例都需要对象的集合,因此,假设我们有一个Student类型的对象的集合,就像我们前面定义的那样:

学生集合

 List students = getStudentObjectCollection(); 

我们从添加到Collection接口的新方法stream()开始。 由于所有集合都“扩展”集合,因此所有Java集合都继承了此方法:

学生流

 List students = getStudentObjectCollection();  Stream stream = students.stream(); // a stream of student objects 

尽管看起来很像,但Stream接口不是另一种常规的集合类型。 我们可以将Stream视为“数据流”抽象,它使我们能够转换或操纵其包含的数据。 与我们在Java中研究过的其他集合不同,Stream不允许我们直接访问其包含的元素。 尽管如果您想访问元素,我们总是可以将流转换为Java中的集合之一并实现我们的目的。

出于演示目的,我们将看到如果我们必须计算students集合中有多少个奇数ID对象,我们的代码将是什么样子。 首先,让我们看看如何在不使用流的情况下完成此操作:

计数奇数

 long count = 0 ;  List students = getStudentObjectCollection();  for (Student s : students) { if (s.getId() % 2 == 1 ) { count++; }  } 

使用for循环,我们创建了一个计数器,每次在学生列表中遇到奇数ID时,该计数器都会递增。 对于一个非常简单的任务,我们已经数百次编写了这种类型的代码,它跨越多行。

我们也可以在一行中使用Stream编写完全相同的代码:

使用流

 List students = getStudentObjectCollection();  long count = students.stream().filter(student -> student.getId() % 2 == 1 ).count(); 

这看起来比以前的for循环方法干净整洁吗? 一切都始于调用stream()方法,该方法将给定的集合转换为Stream,其他所有调用都链接在一起,因为Stream接口中的大多数方法都是在考虑Builder模式的情况下设计的 。 对于那些不习惯使用这种方法进行链接的用户,可能更容易这样可视化:

可视化流

 List students = getStudentObjectCollection();  Stream stream = students.stream();  stream = stream.filter(student -> student.getId() % 2 == 1 );  long count = stream.count(); 

让我们将注意力集中在我们使用的Stream的两种方法中, filter()count()

filter()方法采用要过滤集合的条件,该条件由带一个参数并返回布尔值的lambda表达式表示:

Lambda条件

 student -> student.getId() % 2 == 1 

并非偶然,用于表示该表达式的功能接口filter()方法的参数filter()是谓词接口。 它只有一个抽象方法boolean test(T t)

功能介面

 @FunctionalInterface  public interface Predicate { boolean test(T t); // non-abstract methods here  } 

参数化类型T表示流中元素的类型,即Student对象。 过滤之后,剩下的就是调用count()方法。 没什么大不了的,它只是计算过滤发生后我们流中还剩下多少个对象(除了过滤之外,我们还可以有更多的东西)。 count()方法被视为“终端操作”,在调用该方法后,该流被称为“已消耗”且无法再使用。

6. Lambda表达式的缺点

尽管带有Lambda Expressions的代码看起来非常简洁,但是Lambdas也有一些缺点。 让我们在这里研究其中的一些:

  • 无法处理检查的异常 :引发检查的异常的任何代码都应包装在try-catch语句中。 但是,即使我们这样做,也不总是总是清楚抛出的异常发生了什么。
  • 性能问题 :由于JIT不能始终将forEach() + lambda优化到与普通循环相同的程度,因此Lambda可以在很小程度上影响性能。
  • 调试挑战 :显然,使用Lambdas时,代码并不总是那么简洁。 这使得在代码和可读性方面发生的异常的堆栈跟踪变得有些困难。

尽管Lambda有一些缺点,但是当您编写简洁的代码时,它们仍然是一个很好的伴侣。

7.结论

Java Lambda表达式在所有LISP,Perl,Python以及最新版本的C ++,Objective C,C#和Java 8中都出现(具有不同的语法),但值得注意的是,即使它可以处理传递的函数(或一些借口)作为参数。 它们是具有特定语义的语法元素,并且这些语义对运行时的要求比C所设计的要高。

在本课中 ,我们可以阅读有关Lambda表达式的更多信息,它与功能接口有很深的联系,还演示了将并行流与Lambda表达式配合使用的性能比较,并加深了对Lambda表达式如何与功能接口一起使用并可以在简单语句中使用的理解。利用多核操作系统提供的并行性,而无需了解幕后工作的API。

上次更新时间为2020年2月17日

翻译自: https://www.javacodegeeks.com/lambda-expressions-java-tutorial.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/345644.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

随机过程

第一部分&#xff1a;为什么要研究随机过程&#xff1f; 人类认识世界的历史&#xff0c;就是一认识和描绘各种运动的历史&#xff0c;从宏观的天体运动到分子的运动&#xff0c;到人心理的运动-我们通称为变化&#xff0c;就是一个东西随时间的改变。 人们最成功的描绘运动的…

java steam_Java,Steam控制器和我

java steam您是否想过是否可以将现有的东西用于新的东西&#xff1f; 我看了一些所谓的“蒸汽控制器”&#xff08;从现在开始为SC&#xff09;的镜头&#xff0c;并看着我的游戏手柄。 问我自己是否有可能以类似蒸汽的方式使用它&#xff0c;我找到了一些Java库并创建了一个项…

如何在STS中创建Spring Boot项目

你好朋友&#xff0c; 在本教程中&#xff0c;我们将逐步介绍如何在STS&#xff08;Spring工具套件&#xff09;中创建Spring Boot项目。 步骤1&#xff1a; 如果您尚未从相应的操作系统的以下链接中下载Spring工具套件&#xff0c;请下载&#xff1a; https://spring.io/to…

置信区间(confidence interval)

要理解置信区间&#xff0c;就要从统计学最基本最核心的思想去思考&#xff0c;那就是用样本估计总体。 置信水平(Confidence level)是指总体参数值落在样本统计值某一区内的概率&#xff1b; 而置信区间&#xff08;confidence interval&#xff09;是指在某一置信水平下&…

卫星运动轨道

地球的自转轴与地球公转平面并不垂直&#xff0c;因此&#xff0c;黄道面和地球赤道面并不重合&#xff0c;而是有大约23.4的夹角&#xff0c;通常被称为黄赤交角&#xff0c;也称为转轴倾角、倾角&#xff08;obliquity&#xff09;或轴交角&#xff08;axial inclination&…

随机过程:指数分布、泊松过程、更新过程(renewal process)+大数定律

笔记主要基于中文版《应用随机过程 Introduction to Probability Models 》(Sheldon M. Ross)&#xff0c;只有非常少的一部分是我自己的注解。写这个笔记的目的是自己复习用&#xff0c;阅读需要一定的微积分和概率论基础。本人为初学者&#xff0c;且全部为自学&#xff0c;如…

接受拒绝采样(Acceptance-Rejection Sampling)

我们所说的抽样&#xff0c;其实是指从一个概率分布中生成观察值&#xff08;observations&#xff09;的方法。而这个分布通常是由其概率密度函数&#xff08;PDF&#xff09;来表示的。而且&#xff0c; 即使在已知PDF的情况下&#xff0c;让计算机自动生成观测值也不是一件容…

gradle文件不识别_识别Gradle约定

gradle文件不识别通过约定进行配置具有许多优点&#xff0c;尤其是在简洁方面&#xff0c;因为开发人员不需要显式配置通过约定隐式配置的内容。 但是&#xff0c;在利用约定进行配置时&#xff0c;需要注意这些约定。 这些约定可能已记录在案&#xff0c;但是当我可以编程方式…

Telesat、OneWeb及SpaceX三个全球宽带低轨卫星星座系统的技术对比

编者按&#xff1a;本文来自微信公众号“卫星与网络”&#xff08;ID&#xff1a;satnetdy&#xff09;&#xff0c;作者Inigo del Portilloa,*, Bruce G. Cameronb, Edward F. Crawleyc&#xff0c;编译 刘帅军、胡月梅&#xff08;中科院软件所&#xff09;&#xff0c;36氪经…

腾讯人均每月薪酬成本超8万元,员工总数首次超10万

11月10日&#xff0c;腾讯在23岁“生日”即将到来之际发布2021年第三季度业绩报告。财报显示&#xff0c;第三季度腾讯总收入为人民币1424亿元&#xff08;220亿美元 &#xff09;&#xff0c;同比增长13%&#xff1b;净利润(Non-IFRS&#xff09;317.5亿元&#xff0c;同比减少…

低轨卫星通信系统发展综述

最近几年低轨移动通信领域风起云涌&#xff0c;Iridium、OneWeb、Boeing、SpaceX 这些商业航天的 独角兽 都先后实施或宣布自己在这一领域雄心勃勃的计划。 通信、导航和遥感是卫星应用领域的三驾马车&#xff0c;简称 通导遥。鉴于任务特性&#xff0c;通信卫星和导航卫星通常…

创建您的第一个servlet

在本教程中&#xff0c;我们将学习如何使用Servlet创建非常基本的Web应用程序。 Servlet是一类&#xff0c;扩展了服务器拦截和响应传入请求的功能。 Servlet是一个Web组件&#xff0c;可在服务器上进行编写&#xff0c;构建和部署&#xff0c;以创建动态Web页面。 首先&…

StarLink星座最新动态及星间组网动态路由探讨

StarLink星座最新动态及星间组网动态路由探讨 2020-06-24 11:50 StarLink星座最新动态及星间组网动态路由探讨 作者 | 刘帅军、徐帆江、刘立祥、范媛媛、王大鹏 &#xff08;中国科学院软件研究所&#xff0c;天基综合信息系统重点实验室&#xff09; 一 概述 自2020年6月开…

累计分布函数CDF、互补累计分布函数CCDF、期望Expection

1 CDF 累积分布函数(Cumulative Distribution Function&#xff0c;CDF)&#xff0c;又叫分布函数&#xff0c;是概率密度函数的积分&#xff0c;能完整描述一个实随机变量X的概率分布。一般以大写CDF标记,&#xff0c;与概率密度函数probability density function&#xff08…

markov chain, MRP MDP

在强化学习中&#xff0c;马尔科夫决策过程&#xff08;Markov decision process, MDP&#xff09;是对完全可观测的环境进行描述的&#xff0c;也就是说观测到的状态内容完整地决定了决策的需要的特征。几乎所有的强化学习问题都可以转化为MDP。本讲是理解强化学习问题的理论基…

(网络)流和会话

流:指具有相同五元组(源IP,源端口,目的IP,目的端口,协议)的所有包 会话:指由双向流组成的所有包(源和目的互换)

Filtration, σ-algebras

1. Filtration filtration在钱敏平老师和龚光鲁老师的《随机过程论》中直接称其为非降的KaTeX parse error: Undefined control sequence: \sigmma at position 1: \̲s̲i̲g̲m̲m̲a̲代数族。如图。 一般叫σ\sigmaσ-代数流或σ\sigmaσ-域流 在鞅论中的花体FtF_tFt​&…

gradle 命令行_Gradle命令行便利

gradle 命令行在我的《用Gradle构建Java的gradle tasks 》一文中&#xff0c;我简要提到了使用Gradle的“ gradle tasks ”命令来查看特定Gradle构建的可用任务。 在这篇文章中&#xff0c;我将对这一简短提及进行更多的扩展&#xff0c;并查看一些相关的Gradle命令行便利。 Gr…

怎样更好地理解并记忆泰勒展开式

本段的核心思想是仿造。当我们想要仿造一个东西的时候&#xff0c;无形之中都会按照上文提到的思路&#xff0c;即先保证大体上相似&#xff0c;再保证局部相似&#xff0c;再保证细节相似&#xff0c;再保证更细微的地方相似……不断地细化下去&#xff0c;无穷次细化以后&…

新的DMN编辑器预览

Workbench 7.13.0.Final于10月16日星期二发布&#xff0c;此版本带来了许多有趣的功能和重要的修复程序。 亮点之一是作为技术预览功能的新DMN编辑器&#xff0c;该功能仍在开发中&#xff0c;但您可以开始使用。 在本文中&#xff0c;您将学习如何启用DMN编辑器预览&#xff…