对象作为参数示例java_功能Java示例 第6部分–用作参数

对象作为参数示例java

这是称为“ 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

在这种情况下,我们必须将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

对象作为参数示例java

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

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

相关文章

为什么Java在后来的版本中,给接口增加了默认方法、静态方法、私有方法?

在业务发展过程中&#xff0c;发现某个功能是所有子类都需要的&#xff0c;那么我们该怎么办&#xff1f;给接口增加抽象方法吗&#xff1f;如果直接在接口增加抽象方法&#xff0c;那么所有已经写好的具体实现类就会报编译错误。因为具体类必须实现接口的全部抽象方法。这样就…

linux 源码包安装mysql_LINUX下源码包安装mysql

tar xvfz mysql-5.1.50-linux-i686-glibc23.tar.gz -C /usr/local/解压到的目录cd /usr/local/[rootdrbd2 local]# ln -sv mysql-5.1.50-linux-i686-glibc23 mysql(创建连接)groupadd mysql 添加mysql的用户组useradd -g mysql mysql 增加mysql用户并把其加入mysql组cd mysql[r…

jdk 安全属性_系统属性的JDK 12 Javadoc标记

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

Lambda表达式的学习

文章目录函数式编程思想函数式接口Lambda 的使用前提Lambda 的标准格式Lambda 极简特性演示代码函数式编程思想 与面向对象编程思想的区别&#xff1a; 1.面向对象的思想是做一件事情&#xff0c;找一个能解决这个事情的对象&#xff0c;调用对象的方法来完成事情 2.函数式编程…

公证服务信息_使用多个公证员提高网络吞吐量

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

java判断手机号_java使用正则表达式判断手机号的方法示例

本文实例讲述了java使用正则表达式判断手机号的方法。分享给大家供大家参考&#xff0c;具体如下&#xff1a;要更加准确的匹配手机号码只匹配11位数字是不够的&#xff0c;比如说就没有以144开始的号码段&#xff0c;故先要整清楚现在已经开放了多少个号码段&#xff0c;国家号…

Spring JDBC的学习

文章目录介绍使用步骤常用方法示例代码介绍 所谓 Spring JDBC&#xff0c;是 Spring 框架对 JDBC 的简单封装。提供了一个 JdbcTemplate 对象简化 JDBC 的开发。 使用步骤 步骤&#xff1a; 1.导入相关的 jar 包 2.创建 JdbcTemplate 对象&#xff0c;依赖于数据源 DataSo…

编写脚本电脑怎么编写界面_在任何无法理解的情况下,请编写脚本

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

stack java实现_Stack (堆栈)使用JAVA实现

代码如下&#xff1a;import java.util.AbstractList;import java.util.Arrays;import java.util.Iterator;import java.util.List;import java.util.RandomAccess;public class Stack extends AbstractListimplements List, RandomAccess, Cloneable, java.io.Serializable{pr…

HTML特殊符号/特殊字符

注&#xff1a;实体名称大小写敏感 点击查看更多特殊字符

访客模式 无痕模式 区别_旧访客设计模式的新生活

访客模式 无痕模式 区别介绍 访客 [1、2]是众所周知的经典设计模式。 有很多资源对其进行了详细说明。 在不深入研究实现的情况下&#xff0c;我将简要提醒一下该模式的概念&#xff0c;解释其优点和缺点&#xff0c;并提出一些可以使用Java编程语言轻松应用于其的改进。 古典…

HTML的图片标签img的图片地址

使用相对路径&#xff0c;例如&#xff0c;当前目录中的子目录 image 中的图片 1.jpg&#xff0c;可以写成 ./image/1.jpg&#xff0c;其中 ./ 表示当前目录&#xff0c;可以省略不写。 <img src"image/1.jpg"/>实际上HTML文档数据写入到 Response 中时&#…

java swing 左上角图标_科学网—Matlab: 学习GUI(修改窗口左上角图标而不warning) - 刘磊的博文...

网上常用的方法&#xff1a;if ~isdeployednewIconjavax.swing.ImageIcon(.piciap.jpg);elsenewIconjavax.swing.ImageIcon(iap.jpg);endjFrame get(hObject,javaframe);jFrame.setFigureIcon(newIcon);运行后warning&#xff1a;Warning: figure JavaFrame property will be …

pbfunc外部扩展函数_从外部CorDapp扩展和覆盖流

pbfunc外部扩展函数Corda 4于上周&#xff08;2月21日&#xff09;发布&#xff0c;带来了大量的新功能&#xff0c;使Corda更加令人愉快。 老实说&#xff0c;我有点假设有很多新功能。 我快速浏览了变更日志&#xff0c;主要是看到我的贡献被引用&#xff0c;但是我记得看到很…

表单项标签的input标签的单选框(radio)

<input type"radio" name"gender" value"male"> 男 <input type"radio" name"gender" value"female"> 女1.单选框需要注意的是&#xff0c;如果是属于一组的选项&#xff0c;那么 name 属性的值必须相…

java 6 update 3_Java(TM) 6 Update(java运行环境) V 6.0.450.6 官方版

Java(TM) 6 Update是个JAVA辅助软件&#xff0c;它具备高度的安全性以及跨平台的特性&#xff0c;能让你的电脑或手机运行java程序&#xff0c;用户可使用Java(TM) 6 Update来搭建甚至运行整个ava程序&#xff0c;注意&#xff1a;卸载后JAVA环境的程序将无法运行。过去很可能会…

jep290涉及jdk版本_JDK 9 / JEP 280:字符串串联永远不会相同

jep290涉及jdk版本JEP 280 &#xff08;“ Indify String Concatenation”&#xff09;是与JDK 9一起实现的&#xff0c;根据其“摘要”部分&#xff0c;“更改了javac生成的静态String concatenation字节码序列&#xff0c;以使用对JDK库函数的invokedynamic调用。 ” 通过查看…

label标签/标记

label 标签用于指定表单项的文字描述信息&#xff0c;如下所示&#xff1a; <label for"username">用户名称&#xff1a;</label> <input id"username" name"username">label 标签 的 for 属性指定的值与 input 标签的 id 属…

矩阵累积相乘 java_累积:轻松自定义Java收集器

矩阵累积相乘 javaAccumulative是针对Collector<T, A, R>的中间累积类型A提出的接口Collector<T, A, R>以使定义自定义Java Collector更加容易。 介绍 如果您曾经使用过Java Stream &#xff0c;那么很可能会使用了一些Collector &#xff0c;例如&#xff1a; C…

java socket 传输压缩文件_java基于socket传输zip文件功能示例

本文实例讲述了java基于socket传输zip文件的方法。分享给大家供大家参考&#xff0c;具体如下&#xff1a;服务器端程序&#xff1a;import java.io.*;import java.net.*;import java.io.BufferedInputStream;public class SocketServer {ServerSocket ssnull;Socket snull;Dat…