功能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,一经查实,立即删除!

相关文章

【渝粤题库】陕西师范大学202801 中国古代文学(五) 作业

一、选择题 1、元代出现了一大批杂剧的优秀作家&#xff0c;《析律志》说&#xff1a;“生而倜傥&#xff0c;博学能文&#xff0c;滑稽多智&#xff0c;蕴藉风流&#xff0c;为一时之冠”的作家是&#xff08; &#xff09;&#xff1a; A.王实甫 B.关汉卿 C.马致远 D.高明…

【渝粤题库】陕西师范大学210010 幼儿园管理学 作业(高起专、专升本)

《幼儿园管理学》考评作业 一、填空题 &#xff08;1&#xff09;我国学前教育各级机构权限的划分制度是 、 、 。 &#xff08;2&#xff09;管理内容的五大要素是 、 、 、 、 。 &#xff08;3&#xff09;幼儿园总务工作的特点是 、 、 、 。 &#xff08;4&#xff09;园长…

【渝粤题库】陕西师范大学292281中央银行学Ⅰ作业(高起专)

一、单选题 中央银行之所以成为中央银行&#xff0c;最基本、最重要的标志&#xff08; &#xff09;。 A. 集中存款准备金 B. 集中与垄断货币发行 C. 充当“最后贷款人” D. 代理国库&#xff08; &#xff09;是当今世界上最富于全球代表性和影响力的政府间国际金融组织&…

【渝粤题库】陕西师范大学500004 电动力学

《电动力学》作业 一. 单项选择题 半径为R的均匀磁化介质球&#xff0c;磁化强度为&#xff0c;则介质球的总磁矩为 A. B. C. D. 0 真空中做匀速直线运动的电荷不能产生 A. 电场 B. 磁场 C. 电磁辐射 D. 位移电流 线性介质中磁场的能量密度为 A. B. C. D. 磁场的矢量势是依据…

【渝粤题库】陕西师范大学800001 自然地理学

《自然地理学》作业 一&#xff0e;填空题 &#xff11;&#xff0e;分隔地幔和地核的不连续界面是 。   &#xff12;&#xff0e;摩氏硬度是7的矿物是 &#xff0c;该矿物具有的光泽是 。   &#xff13;&#xff0e;按照地震的成因不同&#xff0c;地震可分为 、 和 三种…

系统属性的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标记 ”中进行了讨论&…

国家开放大学2021春1079高等代数专题研究题目

教育 教育 试卷代号&#xff1a;1079 2021年春季学期期末统一考试 高等代数专题研究 试题 2021年7月 一、单项选择题&#xff08;本题共20分&#xff0c;每小题4分&#xff09; 1.下列运算中&#xff0c;&#xff08; &#xff09;是有理数域Q上的代数运算. A.a。ba B.a。bb …

【渝粤教育】广东开放大学 Android移动开发项目实践 形成性考核 (58)

题库查询系统退出 选择题 题目&#xff1a;下列选项中&#xff0c;属于Android Studio工具中创建项目时选择的按钮的是&#xff08;&#xff09;。 答案&#xff1a; A、Start a new Android Studio project 题目&#xff1a;下列选项中&#xff0c;属于Log类中显示黑色的日志内…

qt服务器获取formdata文件,QT上传(PUT)文件

我正在使用QT将文件上传到Web服务器。QT上传(PUT)文件curl -X POST -H Content-Type:multipart/form-data-H Authorization: Token -F [email protected]_to_upload.txthttps://some.web.site/api/v2/files/contents/我使用大致是这样的QT调用尝试来完成相同的&#xff1a;QHtt…

vue cli3 绑定域名_使用CLI设置WildFly绑定地址并关闭

vue cli3 绑定域名仅使用命令行参数将WildFly绑定到主机名/ IP上非常容易。 我有一个简单的GNU / Linux盒子&#xff0c;可以用它玩各种东西&#xff0c;其中之一就是WildFly。 我使用以下命令开始在特定IP上监听WildFly&#xff1a; $> cd /opt/wildfly/wildfly-8.0.0.Be…

ajax post提交特殊字符,如何利用jQuery post传递含特殊字符的数据

在jQuery中&#xff0c;我们通常利用$.ajax或$.post进行数据传递处理&#xff0c;但这里通常不能传递特殊字符&#xff0c;如&#xff1a;“1、准备页面和控制端代码页面代码如下&#xff1a;$(function() {$("#btnSet").click(function() {var a $("#txtValue…

功能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…

【渝粤教育】广东开放大学 javaweb 动态网站开发技术 形成性考核 (53)

选择题 题目&#xff1a;关于MVC模式说法正确的是_____ 题目&#xff1a;在MVC模式中&#xff0c;核心内容为_____ 题目&#xff1a;下面关于MVC的说法不正确的是_____ 题目&#xff1a;MVC设计模式将应用程序分为_____部分 题目&#xff1a;MVC中的M是指&#xff08;&#xff…

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

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

【渝粤教育】广东开放大学 Python基础与应用 形成性考核 (21)

选择题 题目&#xff1a;Python语言语句块的标记是&#xff08; &#xff09; 题目&#xff1a;Python程序的文件扩展名是&#xff08; &#xff09; 题目&#xff1a;下列运算符中优先级最高的是&#xff08; &#xff09; 题目&#xff1a;Python不支持的数据类型…

带有AWS DynamoDB的反应式Spring Webflux

AWS已经发布了Java版本2的AWS开发工具包 &#xff0c;该开发工具包现在支持针对不同AWS服务的API调用的非阻塞IO。 在本文中&#xff0c;我将探讨如何使用AWS开发工具包2.x的DynamoDB API以及如何使用Spring Webflux堆栈公开响应式端点-这种方式&#xff0c;应用程序是端对端响…

【渝粤教育】广东开放大学 人工智能 形成性考核 (55)

选择题 题目&#xff1a;连接主义认为人工智能的研究方法应采用&#xff08;&#xff09;方法。 题目&#xff1a;图灵曾协助军方破解()的著名密码系统Enigma。 题目&#xff1a;由《大英百科全书》可知,围棋大约源起于公元前()年左右。 题目&#xff1a;下面属于常用的点角破空…

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

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

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

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