功能Java示例 第6部分–用作参数

这是称为“ Functional Java by Example”的系列文章的第6部分。

我在本系列的每个部分中发展的示例是某种“提要处理程序”,用于处理文档。 在前面的部分,我们试图通过移动尽可能多的副作用,如IO,该系统的外部,以使我们的可能的功能。

现在,我们将一些抽象替换为函数,以作为参数传递。

如果您是第一次来,最好是从头开始阅读。 它有助于了解我们从何处开始以及如何在整个系列中继续前进。

这些都是这些部分:

  • 第1部分–从命令式到声明式
  • 第2部分–讲故事
  • 第3部分–不要使用异常来控制流程
  • 第4部分–首选不变性
  • 第5部分–将I / O移到外部
  • 第6部分–用作参数
  • 第7部分–将失败也视为数据
  • 第8部分–更多纯函数

我将在每篇文章发表时更新链接。 如果您通过内容联合组织来阅读本文,请查看我博客上的原始文章。

每次代码也被推送到这个GitHub项目 。

OO型协作者

还记得我们以前留下的东西吗?

class FeedHandler {Webservice webserviceList<Doc> handle(List<Doc> changes) {changes.findAll { doc -> isImportant(doc) }.collect { doc ->createResource(doc).thenApply { resource ->setToProcessed(doc, resource)}.exceptionally { e ->setToFailed(doc, e)}.get()}}private CompletableFuture<Resource> createResource(doc) {webservice.create(doc)}private static boolean isImportant(doc) {doc.type == 'important'}private static Doc setToProcessed(doc, resource) {doc.copyWith(status: 'processed',apiId: resource.id)}private static Doc setToFailed(doc, e) {doc.copyWith(status: 'failed',error: e.message)}}

上面的提要处理程序需要一个“ Web服务”来完成其工作。

请看以下部分,其中使用WebService类型的协作者来基于文档创建资源:

class FeedHandler {Webservice webserviceList<Doc> handle(List<Doc> changes) {changes.collect { doc ->createResource(doc)...}private CompletableFuture<Resource> createResource(doc) {webservice.create(doc)}}

请记住, 作为异常处理机制的一部分,我们将其包装在CompletableFuture ,而不是直接返回资源。

如果我们想要WebService以外的其他资源来创建资源怎么办?

好吧,这是同时变得棘手和容易的地方-OO风格可能与FP风格有些冲突。

您会看到, WebService是一个Java接口,定义如下:

interface Webservice {CompletableFuture<Resource> create(Doc doc)
}

这遵循了Dependency Inversion Principle(DIP) ,它是Robert C. Martin提倡的SOLID设计原则的一部分,该原则(其中包括)说:

抽象不应依赖细节。 细节应取决于抽象。

WebService已经是任何类型的Webservice 实现的抽象。 因此,系统可以具有此接口的多种实现,例如REST实现和SOAP实现:

class RestWebService implements Webservice {@OverrideCompletableFuture<Resource> create(Doc doc) {// do REST communication}
}
class SoapWebService implements Webservice {@OverrideCompletableFuture<Resource> create(Doc doc) {// do SOAP communication}
}

提要处理程序不关心细节 ,它只是想要一些符合WebService接口定义的协定的东西:有一个create方法可以接受Doc并返回CompletableFuture

FeedHandler类具有一个webservice属性,其中包含对WebService的引用。 任何OO开发人员都可以识别这种样式,因为它非常熟悉:所有协作者都存在于属性中,这些属性(通常)是在构造过程中初始化的。

一旦构造了FeedHandler ,就可以通过DI框架或普通的手工方法将传递的WebService实例传递给它-尽管构造函数注入或属性注入。

为了简洁起见,我一直在代码片段中省略了构造函数,但是正如您在测试用例中所看到的那样, 我绝对使用Groovy为我生成的构造函数来传递所有依赖项。

协作者FP风格

好的,如果我们再次戴上Functional Hat,我们将需要重新审视将WebService传递到提要处理程序的方式。

handle方法的签名不提比其他任何东西:文件进去 ,文件出来

class FeedHandler {...List<Doc> handle(List<Doc> changes) {...}}

我不能假定将返回相同的输入 相同的输出 -因为该方法暗中依赖于外的东西:对WebService

好吧,也许我可以控制供稿处理程序的整个创建过程,包括WebService ,但是在方法调用之间可以更改对webservice引用,每次handle使用它时都会产生其他结果。 除非我将其设为不可变的,否则将阻止引用的更新。 我告诉过你可能会很棘手

是否可以像上一期中使用isImportantsetToProcessedsetToFailed方法一样使handle pure

在这种情况下,我们必须将WebService作为参数传递,就像传递文档列表一样。

我们改变

class FeedHandler {Webservice webserviceList<Doc> handle(List<Doc> changes) {...}}

进入

class FeedHandler {List<Doc> handle(List<Doc> changes, Webservice webservice) {...}}

在每次调用handle我们都会传递它需要的所有内容:需要处理的文档和需要使用的Web服务。

由于此方法不再依赖于FeedHandler类中的任何属性,因此我们现在可以使其变为static -将其升级为类级方法。

功能性Java

高阶函数

实际上,我们的handle方法刚刚变成了所谓的“高阶函数”,即接受一个函数或返回一个函数的函数。

因此,回到我最初提出的一个问题: 如果我们想要WebService之外的其他东西来创建资源该怎么办?

它甚至不应该是Web服务吗? 也许我们完全想吃香蕉,一只猴子为我们创造资源?

class Monkey implements Webservice {@OverrideCompletableFuture<Resource> create(Doc doc) {// go bananas! But do create resources plz}
}

看起来很奇怪,不是吗? 对于抽象提要处理程序的需要, WebService接口太具体了。 任何创造资源的东西都会起作用,不是吗?

更好的名称是“ ResourceCreator” ,因此只需重命名接口即可。

旧:

interface Webservice {CompletableFuture<Resource> create(Doc doc)
}

新:

interface ResourceCreator {CompletableFuture<Resource> create(Doc doc)
}

具有create方法的ResourceCreator接口; 多么合身! 现在任何东西都可以实现此接口,并且提要处理程序甚至不在乎它是Web服务,猴子还是霍比特人。

新方法签名:

class FeedHandler {List<Doc> handle(List<Doc> changes, ResourceCreator creator) {...}}

完美的抽象!

功能抽象

在Java中,我们将只有一种抽象方法接口称为功能接口 。 我们的ResourceCreator符合此描述; 它只有一个抽象方法create

Java的java.util.function程序包具有许多这样的功能接口-它们每个都有一个已定义的目的:

  • Consumer表示一个接受参数且不返回任何值的函数
  • Supplier表示不接受任何参数的函数,仅返回结果
  • Function表示接受一个参数并返回结果的函数
  • …和更多

这意味着,我们不需要每次都需要一个函数“接受一个参数并返回结果”时就定义一个特定的接口,例如ResourceCreatorFunction已经是一个我们可以利用的接口!

这就是Java 8中的Function (简体)的样子:

interface Function<T,R> {R apply(T t);
}

这就是ResourceCreator现在的样子:

interface ResourceCreator {CompletableFuture<Resource> create(Doc doc)
}

您将看到,如果满足以下条件,我们可以用Function完全替代ResourceCreator

  • Doc代替R
  • T型替代CompletableFuture
  • 用方法替代调用create apply

我们可以完全擦除ResourceCreator界面!

新方法签名将变为:

class FeedHandler {List<Doc> handle(List<Doc> changes,Function<Doc, CompletableFuture<Resource>> creator) {...}}

我们取得了什么成就?

  • 我们现在可以传递任何要handle 函数 ,该函数需要一个Doc并产生一个CompletableFuture —这就是feed处理程序正常工作所需的全部。
  • 正如您现在可能已经注意到的,函数编程处理了很多函数 。 一个函数可以采用另一个函数,也可以返回一个函数。
  • 从Java 8开始,我们已经准备好了很多功能接口。 每个开发人员都可以以标准化的方式与他们合作,因此最好查看它们是否适合您的用例和API,并尽可能重用它们。 他们每个人都有泛型类型(如TR可以由你来表明发生什么,什么来的函数的输出 )。

现在,完整的代码如下所示:

class FeedHandler {List<Doc> handle(List<Doc> changes,Function<Doc, CompletableFuture<Resource>> creator) {changes.findAll { doc -> isImportant(doc) }.collect { doc ->creator.apply(doc).thenApply { resource ->setToProcessed(doc, resource)}.exceptionally { e ->setToFailed(doc, e)}.get()}}private static boolean isImportant(doc) {doc.type == 'important'}private static Doc setToProcessed(doc, resource) {doc.copyWith(status: 'processed',apiId: resource.id)}private static Doc setToFailed(doc, e) {doc.copyWith(status: 'failed',error: e.message)}}

现在就这样! 下次,我们将处理故障数据。

如果您有任何意见或建议,我很想听听他们的意见!

翻译自: https://www.javacodegeeks.com/2018/12/functional-java-functions-parameters.html

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

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

相关文章

系统属性的JDK 12 Javadoc标记

JDK 12 Early Access Build 20 &#xff08; 2018/11/15 &#xff09;可用&#xff0c;可以用来试用新的Javadoc标签{systemProperty} 。 新的{systemProperty} Javadoc标记在core-libs-dev邮件列表消息“ FYI&#xff1a;用于文档系统属性的新javadoc标记 ”中进行了讨论&…

功能Java示例 第5部分–将I / O移到外部

这是称为“ Functional Java by Example”的系列文章的第5部分。 在上一部分中&#xff0c;我们停止了对文档的变异&#xff0c;并返回了数据的副本。 现在&#xff0c;我们需要移走一些I / O。 如果您是第一次来&#xff0c;最好是从头开始阅读。 它有助于了解我们从何处开始…

实现打包后修改服务器接口地址,vue打包之后生成一个配置文件修改接口

我们的vue代码打包上传到服务器之后&#xff0c;生成一个配置文件&#xff0c;里面可以配置域名或其它什么字段之类的&#xff0c;这样以后换了域名&#xff0c;只修改这个配置文件即可。第一步&#xff1a;安装generate-asset-webpack-plugin插件npm install --save-dev gener…

我的世界无人维护的服务器,我的世界:如何进入9年无人管理的2B2T?全球最大战争服务器!...

原标题&#xff1a;我的世界&#xff1a;如何进入9年无人管理的2B2T&#xff1f;全球最大战争服务器&#xff01;2b2t因混乱和9年无人管理而闻名于世&#xff0c;目前是全球最大的《我的世界》战争服务器&#xff0c;同时又是第二大Minecraft古老的服务器。​最近很多小伙伴都在…

集团bim对集团项目服务器,BIM再添一员,五洋建设集团BIM项目组举行成立仪式

BIM项目组的成立&#xff0c;标志着五洋建设集团自此迈入了可视化数字建筑信息模型的阵营&#xff0c;掀开了五洋建设集团设计、施工一体化服务新的一页。随后&#xff0c;五洋建筑设计院院长金杭杭主持召开了工作会议。金杭杭院长在致辞中表示&#xff0c;由建设集团技术管理中…

java pojo使用_在POJO中使用ThreadLocal进行Java嵌套事务

java pojo使用大多数嵌套事务是使用EJB实现的&#xff0c;现在我们尝试在POJO上实现嵌套事务。 在这里&#xff0c;我们使用了ThreadLocal的功能。 了解嵌套事务 事务可以嵌套在另一个内部。 因此&#xff0c;内部事务或外部事务可以回滚或提交&#xff0c;而不会影响其他事务…

Java开发人员应该知道的5大Spring Boot功能

您可能已经听说过Spring Boot&#xff0c;这是用不到140个字符创建一个Spring Web应用程序的神奇力量&#xff0c;可以在一条推文中编写这些字符&#xff0c;但这到底意味着什么&#xff1f; 哪些功能可以使Spring Boot具有如此强大的功能并使Spring应用程序开发如此容易&#…

java 使用本机代理_Java与本机代理–他们所做的强大功能

java 使用本机代理在安装代理之前应了解的内容及其对代码的影响 在构建可伸缩的服务器端应用程序时&#xff0c;我们花费大量时间思考如何在生产中监视&#xff0c;操作和更新代码。 已经开发出一种新的工具来帮助Java和Scala开发人员做到这一点。 它们中的许多都是建立在最强大…

在任何无法理解的情况下,请编写脚本

脚本编写是使您的应用程序在运行时就可以根据客户需求进行调整的最流行的方法之一。 与往常一样&#xff0c;此方法不仅带来好处&#xff0c;例如&#xff0c;在灵活性和可管理性之间存在众所周知的折衷方案。 本文不是从理论上讨论优缺点的文章之一&#xff0c;而是从实践上展…

下载anaconda时出现“Please make sure you are connected to the internet”警告

如题&#xff0c;在anaconda下载过程中下载VScode时出现下图的警告。 百度翻译&#xff1a; 顺着图中指定文件路径&#xff0c;找到vscode_inst.py.log文件&#xff08;注&#xff1a;有些人ProgramData文件夹可能找不到&#xff0c;打开任意文件夹&#xff0c;点击查看&…

多个公证员提高网络吞吐量

您是否需要非常高吞吐量的Corda网络&#xff1f; 网络的吞吐量是否稳定&#xff1f; 您是否已经从其他领域挤出了所有可能的表现&#xff1f; 如果您对这些问题的回答是“是”&#xff0c;那么我可能会为您提供一些有用的信息。 我列出了这些问题&#xff0c;以减少您过早优化C…

初识FPGA(搬运)

原文链接1原文链接2 fpga简介 FPGA&#xff08;Field&#xff0d;Programmable Gate Array&#xff09;&#xff0c; 即现场可编程门阵列&#xff0c;它是在PAL&#xff08;可编程阵列逻辑&#xff09;、GAL&#xff08;通用阵列逻辑器件&#xff09;、CPL&#xff08;复杂可…

蓝桥杯小白系列之汇编点亮led灯

蓝桥杯小白系列之汇编点亮led灯 1、源代码 2、逐条分析 (1)ORG 0000H (2)START: 伪指令,编译器可识别,单片机不可识别,可以随便起。 (3)蜂鸣器设置 蓝桥板插电以后蜂鸣器常会自动响起,故在程序前提前加上如下代码,让蜂鸣器不响: mov P2,#0A0H mov P0,#000H 参照…

带有Oracle Digital Assistant和Fn Project的会话式UI。 第二部分

在上一篇文章中&#xff0c;我使用Oracle Digital Assistant为FlexDeploy实现了一个对话式UI。 今天&#xff0c;我将用Fn Flow丰富它&#xff0c;以便聊天机器人接受发行名称而不是ID来创建快照。 完成后&#xff0c;对话听起来会更加自然&#xff1a; … “您可以建立快照吗…

小白系列之51单片机的入门速成法

51单片机的入门速成法 01-经典51内核资源全览浓缩图 02-重要外设特殊功能寄存器概览 03-程序开发流程与设计要点 04-三大外设的开发与可重用代码 05-应用程序设计入门一例通 01-经典51内核资源全览浓缩图 <1> 四组8位并行I/O端口&#xff1a; P0端口&#xff1a;PC门&a…

蓝桥杯基础模块1:LED跑马灯

一、模块题目 二、原理简述 1、74HC138:三八译码器(3个输入,8个输出) 2、74HC573:锁存器(20个引脚,D1D8是数据输入端,Q1Q8是数据输出端&#

如何使用Hibernate将PostgreSQL枚举映射到JPA实体属性

介绍 开源的hibernate-types项目允许您映射JSON&#xff0c;ARRAY&#xff0c; YearMonth &#xff0c; Month或数据库特定的列&#xff08;例如INET地址&#xff09;。 在本文中&#xff0c;我们将看到使用JPA和Hibernate时如何将PostgreSQL Enum类型映射到Java数组。 Maven…

蓝桥杯基础模块2:蜂鸣器继电器

一、模块题目 二、原理简述 1、74HC138(参见模块1) 2、74HC02(参见模块1) 3、74HC573(参见模块1)

蓝桥杯基础模块3_1:数码管静态显示

一、模块题目 二、原理简述 1、数码管 CT107D单片机综合实训平台上使用的数码管是F3461BH(倒数第二个字母是A则共阴,是B则共阳)。 F3461BH是一个4位8段的数码管,其中a、b、c、d、e、f、g、dp引脚分别对应8个段码,该8个引脚通过74HC573锁存器与单片机的P0端口相连。另外有…

蓝桥杯基础模块3_2:数码管动态显示

一、模块题目 二、原理简述 动态显示的基本原理与实现思路(转载) 动态显示实质上就是轮流点亮单个数码管实现多位数码管整体显示的效果。在轮流显示过程中,每位数码管点亮时间为1~2ms,由于人的视觉暂留现象及发光二极管的余辉效应,尽管实际上各位数码管并非同时点亮,但…