接口文档示例_功能Java示例第8部分–更多纯函数

接口文档示例

接口文档示例

这是第8部分,该系列的最后一部分称为“ Functional Java by Example”。

我在本系列的每个部分中发展的示例是某种“提要处理程序”,用于处理文档。 在上一期文章中,我们已经使用Vavr库看到了一些模式匹配,并且还将故障也视为数据,例如,采用替代路径并返回到功能流程。

在本系列的最后一篇文章中,我将功能发挥到了极致:一切都变成了功能。

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

这些都是这些部分:

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

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

最大化运动部件

您可能已经听过Micheal Feathers的以下短语:

OO通过封装运动部件使代码易于理解。 FP通过最大程度地减少运动部件来使代码易于理解。

好的,让我们稍微忘记上一期中的故障恢复,然后继续使用下面的版本:

 FeedHandler { class FeedHandler { 
  List<Doc> handle(List<Doc> changes,

    Function<Doc, Try<Resource>> creator) {

    changes

      .findAll { doc -> isImportant(doc) }

      .collect { doc ->

        creator.apply(doc)

        }.map { resource ->

          setToProcessed(doc, resource)

        }.getOrElseGet { e ->

          setToFailed(doc, e)

        }

      }

  }

  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

    )

  }
 }

替换为功能类型

我们可以使用对函数接口类型的变量(例如PredicateBiFunction的引用来替换每种方法。

A)我们可以替换一个接受1个参数并返回boolean的方法

 private static boolean isImportant(doc) {

  doc.type == 'important'
 }

谓词

 private static Predicate<Doc> isImportant = { doc ->

  doc.type == 'important'
 }

B),我们可以替换一个接受2个参数并返回结果的方法

 private static Doc setToProcessed(doc, resource) {

  ...
 }
 private static Doc setToFailed(doc, e) {

  ...
 }

具有双功能

 private static BiFunction<Doc, Resource, Doc> setToProcessed = { doc, resource ->

  ...
 }
 private static BiFunction<Doc, Throwable, Doc> setToFailed = { doc, e ->

  ...
 }

为了实际调用封装在(Bi)Function中的逻辑,我们必须对其调用apply 。 结果如下:

 FeedHandler { class FeedHandler { 
  List<Doc> handle(List<Doc> changes,

    Function<Doc, Try<Resource>> creator) {

    changes

      .findAll { isImportant }

      .collect { doc ->

        creator.apply(doc)

        .map { resource ->

          setToProcessed.apply(doc, resource)

        }.getOrElseGet { e ->

          setToFailed.apply(doc, e)

        }

      }

  }

  private static Predicate<Doc> isImportant = { doc ->

    doc.type == 'important'

  }

  private static BiFunction<Doc, Resource, Doc> setToProcessed = { doc, resource ->

    doc.copyWith(

      status: 'processed' ,

      apiId: resource.id

    )

  }

  private static BiFunction<Doc, Throwable, Doc> setToFailed = { doc, e ->

    doc.copyWith(

      status: 'failed' ,

      error: e.message

    )

  }
 }

将所有输入移至功能本身

我们将所有内容移至方法签名,以便FeedHandler的handle方法的调用者可以提供自己的那些功能的实现。

方法签名将更改为:

 List<Doc> handle(List<Doc> changes,

  Function<Doc, Try<Resource>> creator)

 List<Doc> handle(List<Doc> changes,

  Function<Doc, Try<Resource>> creator,

  Predicate<Doc> filter,

  BiFunction<Doc, Resource, Doc> successMapper,

  BiFunction<Doc, Throwable, Doc> failureMapper)

其次,我们将重命名原始(静态)谓词BiFunction变量

  • isImportant
  • setToProcessed
  • setToFailed

转换为类顶部的新常量,以反映它们的新作用。

  • DEFAULT_FILTER
  • DEFAULT_SUCCESS_MAPPER
  • DEFAULT_FAILURE_MAPPER

客户端可以完全控制是否将默认实现用于某些功能,或者何时需要接管自定义逻辑。

例如,当仅需要定制故障处理时,可以这样调用handle方法:

 BiFunction<Doc, Throwable, Doc> customFailureMapper = { doc, e ->

  doc.copyWith(

    status: 'my-custom-fail-status' ,

    error: e.message

  )
 }
 new FeedHandler().handle(...,

  FeedHandler.DEFAULT_FILTER,

  FeedHandler.DEFAULT_SUCCESS_MAPPER,

  customFailureMapper

  )

如果您的语言支持,则可以通过分配默认值来确保客户端实际上不必提供每个参数。 我正在使用支持将默认值分配给方法中的参数的Apache Groovy

 List<Doc> handle(List<Doc> changes,

  Function<Doc, Try<Resource>> creator,

  Predicate<Doc> filter = DEFAULT_FILTER,

  BiFunction<Doc, Resource, Doc> successMapper = DEFAULT_SUCCESS_MAPPER,

  BiFunction<Doc, Throwable, Doc> failureMapper = DEFAULT_FAILURE_MAPPER)

在我们将应用另一个更改之前,请看一下代码:

 FeedHandler { class FeedHandler { 
  private static final Predicate<Doc> DEFAULT_FILTER = { doc ->

    doc.type == 'important'

  }

  private static final BiFunction<Doc, Resource, Doc> DEFAULT_SUCCESS_MAPPER = { doc, resource ->

    doc.copyWith(

      status: 'processed' ,

      apiId: resource.id

    )

  }

  private static final BiFunction<Doc, Throwable, Doc> DEFAULT_FAILURE_MAPPER = { doc, e ->

    doc.copyWith(

      status: 'failed' ,

      error: e.message

    )

  }

  List<Doc> handle(List<Doc> changes,

                   Function<Doc, Try<Resource>> creator,

                   Predicate<Doc> filter = DEFAULT_FILTER,

                   BiFunction<Doc, Resource, Doc> successMapper = DEFAULT_SUCCESS_MAPPER,

                   BiFunction<Doc, Throwable, Doc> failureMapper = DEFAULT_FAILURE_MAPPER) {

    changes

      .findAll { filter }

      .collect { doc ->

        creator.apply(doc)

        .map { resource ->

          successMapper.apply(doc, resource)

        }.getOrElseGet { e ->

          failureMapper.apply(doc, e)

        }

      }

  }
 }

介绍两者

您是否注意到以下部分?

 .collect { doc ->

  creator.apply(doc)

  .map { resource ->

    successMapper.apply(doc, resource)

  }.getOrElseGet { e ->

    failureMapper.apply(doc, e)

  }
 }

请记住, creator的类型是

Function<Doc, Try<Resource>>

表示它返回一个Try 。 我们在第7部分中介绍了Try ,它是从Scala等语言中借用的。

幸运的是, collect { doc的“ doc”变量仍在传递给我们需要它的successMapperfailureMapper范围内,但是Try#map的方法签名(接受一个Function )与我们的successMapper (即一个BiFunctionTry#getOrElseGet也是Try#getOrElseGet ,它也只需要一个Function

从Try Javadocs:

  • map(Function <?super T,?extended U>映射器)
  • getOrElseGet(Function <?super Throwable ,?

简而言之,我们需要从

  1. BiFunction <文档,资源,文档> successMapper
  2. BiFunction <文档,Throwable,文档> failureMapper

  1. 函数<资源,文档> successMapper
  2. 函数<Throwable,Doc> failureMapper

同时仍然可以将原始文档作为输入

让我们介绍两个简单的类型,它们封装了2个BiFunction的2个参数:

 class CreationSuccess {

  Doc doc

  Resource resource
 }
 class CreationFailed {

  Doc doc

  Exception e
 }

我们将论点从

  1. BiFunction <文档,资源,文档> successMapper
  2. BiFunction <文档,Throwable,文档> failureMapper

改为功能

  1. 函数<CreationSuccess,Doc> successMapper
  2. 函数<CreationFailed,Doc> failureMapper

现在, handle方法如下所示:

 List<Doc> handle(List<Doc> changes,

                 Function<Doc, Try<Resource>> creator,

                 Predicate<Doc> filter,

                 Function<CreationSuccess, Doc> successMapper,

                 Function<CreationFailed, Doc> failureMapper) {

  changes

    .findAll { filter }

    .collect { doc ->

      creator.apply(doc)

      .map(successMapper)

      .getOrElseGet(failureMapper)

    }
 }

……但是还行不通

Try使mapgetOrElseGet需要分别。 一种

  • 函数<资源,文档> successMapper
  • 函数<Throwable,Doc> failureMapper

这就是为什么我们需要将其更改为另一个著名的FP结构,称为Either

幸运的是Vavr也有Either 。 它的Javadoc说:

任一代表两种可能的值。

通常使用Either类型来区分正确(“正确”)或错误的值。

它变得非常抽象:

一个Either可以是Either.Left或Either.Right。 如果给定的Either是Right并投影到Left,则Left操作对Right值没有影响。 如果给定的Either是Left并投影到Right,则Right操作对Left值没有影响。 如果将“左”投影到“左”或将“右”投影到“右”,则操作会生效。

让我解释一下上面的神秘文档。 如果我们更换

Function<Doc, Try<Resource>> creator

通过

Function<Doc, Either<CreationFailed, CreationSuccess>> creator

我们将CreationFailed分配给“ left”参数,按照惯例通常会保留错误(请参见Either上的Haskell文档), CreationSuccess是“ right”(和“正确”)值。

在运行时,该实现用于返回Try ,但现在可以返回Either.Right ,以防成功,例如

 return Either.right(

  new CreationSuccess(

    doc: document,

    resource: [id: '7' ]

  )
 )

Either.Left ,但发生故障时除外-两者都包括原始文档。 是。

因为现在类型最终匹配,所以我们终于压扁了

 .collect { doc ->

  creator.apply(doc)

  .map { resource ->

    successMapper.apply(doc, resource)

  }.getOrElseGet { e ->

    failureMapper.apply(doc, e)

  }
 }

进入

 .collect { doc ->

  creator.apply(doc)

  .map(successMapper)

  .getOrElseGet(failureMapper)
 }

现在, handle方法如下所示:

 List<Doc> handle(List<Doc> changes,

                 Function<Doc, Either<CreationFailed, CreationSuccess>> creator,

                 Predicate<Doc> filter,

                 Function<CreationSuccess, Doc> successMapper,

                 Function<CreationFailed, Doc> failureMapper) {

  changes

    .findAll { filter }

    .collect { doc ->

      creator.apply(doc)

      .map(successMapper)

      .getOrElseGet(failureMapper)

    }
 }

结论

我可以说我已经实现了开始时设定的大多数目标:

  • 是的,我设法避免了重新分配变量
  • 是的,我设法避免了可变数据结构
  • 是的,我设法避免了状态(至少在FeedHandler中)
  • 是的,我设法偏爱函数(使用某些Java内置函数类型和某些第三方库Vavr)

我们已经将所有内容移到了函数签名,以便FeedHandler的handle方法的调用者可以直接传递正确的实现。 如果从头到尾回顾原始版本,您会注意到在处理更改列表时,我们仍然承担所有责任:

  • 通过某些条件过滤文档列表
  • 为每个文档创建资源
  • 成功创建资源后执行一些操作
  • 无法创建资源时执行其他操作

然而,在第一部分中,这些责任是势在必行写出来,for语句声明,都在一个大聚集在一起handle方法。 现在,最后,每个决定或动作都由具有抽象名称的函数表示,例如“过滤器”,“创建者”,“ successMapper”和“ failureMapper”。 实际上,它成为一个高阶函数,以多个函数之一作为参数。 提供所有参数的责任已被移至客户端的上一级。 如果查看GitHub项目,您会注意到,对于这些示例,我不得不不断更新单元测试。

有争议的部分

实际上,如果不需要,我可能不会编写我的(Java)商业代码,例如FeedHandler类在传递通用Java函数类型(即FunctionBiFunctionPredicateConsumerSupplier )方面的使用方式所有这些极端的灵活性。 所有这些都是以可读性为代价的。 是的,Java是一种静态类型的语言,因此,使用泛型时,必须在所有类型参数中明确使用一种语言,从而导致以下功能的签名困难:

 handle(List<Doc> changes,
 Function<Doc, Either<CreationFailed, CreationSuccess>> creator,
 Predicate<Doc> filter,
 Function<CreationSuccess, Doc> successMapper,
 Function<CreationFailed, Doc> failureMapper)

在普通JavaScript中,您将没有任何类型,并且您必须阅读文档以了解每个参数的期望。

handle = function (changes, creator, filter, successMapper, failureMapper)

但是,这是一个折衷方案。 Groovy中,也是一个JVM语言,可以让我省略所有的例子类型的信息在这个系列中,甚至允许我使用闭包(象Java lambda表达式)是在Groovy中的函数式编程范式的核心。

更极端的做法是在类级别指定所有类型,以使客户端具有最大的灵活性,以便为不同的FeedHandler实例指定不同的类型。

 handle(List<T> changes,
 Function<T, Either<R, S>> creator,
 Predicate<T> filter,
 Function<S, T> successMapper,
 Function<R, T> failureMapper)

什么时候合适?

  • 如果您完全控制代码,则在特定上下文中使用它来解决特定问题时,这将过于抽象而无法产生任何收益。
  • 但是,如果我要向全世界(或者在组织内向其他团队或部门使用)开放一个库或框架,并将其用于各种不同的用例,那么我可能不会事先想到,为灵活性而设计可能值得。 让呼叫者决定如何过滤以及成功或失败的构成是明智之举。

最终,上述内容在API设计,是和解耦方面稍有改动,但是在典型的Enterprise(tm)Java项目中“使一切成为函数”可能需要与您和您的团队成员进行一些讨论。 多年来,一些同事已经习惯了一种更传统,更惯用的代码编写方式。

好的零件

  • 我绝对希望使用不可变的数据结构(和“引用透明性”)来帮助推断我的数据所处的状态。想想Collections.unmodifiableCollection的集合。 在我的示例中,我将Groovy的@Immutable用于POJO,但在普通的Java库(例如ImmutablesAutoValueProject Lombok)中也可以使用。
  • 最大的改进实际上是导致了一种更具功能性的样式:使代码讲故事,这主要是关于分离关注点并适当地命名事物。 在任何编程风格(即使是OO:D)中,这都是一个好习惯,但这确实消除了混乱,并允许引入(纯)函数。
  • 在Java中,我们习惯于以特定方式进行异常处理,以至于像我这样的开发人员很难提出其他解决方案。 诸如Haskell之类的功能语言仅返回错误代码,因为“ Niklaus Wirth认为异常是GOTO的转世,因此省略了它们” 。 在Java中,可以使用CompletableFuture或…
  • 通过引入第三方库(例如Vavr)可在您自己的代码库中使用的特定类型(例如TryEither )可以在很大程度上帮助您启用更多以FP样式编写的选项! 我以流畅的方式编写“成功”或“失败”路径并且可读性很强,这让我非常着迷。

Java不是F#的Scala或Haskell或Clojure,它最初遵循的是面向对象编程(OOP)范例,就像C ++,C#,Ruby等一样,但是在Java 8中引入了lambda表达式并结合了一些很棒的功能之后如今,开放源代码库如今,开发人员绝对可以选择OOP和FP必须提供的最佳元素

做系列的经验教训

我从很早以前就开始了这个系列的讨论。 早在2017年,我发现自己在一段代码中进行了一些FP风格的重构,这启发了我去寻找一系列名为“ Functional Java by Example”的文章的示例。 这成为我在每个批次中一直使用的FeedHandler代码。

那时我已经对所有的代码进行了更改,但是当时我打算写实际的博客文章,我常常想到:“我只是不能展示重构,我必须实际解释一下!” 那就是我为自己设置陷阱的地方,因为在整个过程中,我坐下来写作的时间越来越少。 (任何写过博客的人都知道,简单地分享要点和撰写可理解的英语😉的连贯段落在时间上的区别)

下次当我想做一个系列时,我会回Google吸取一些经验教训:

  1. 如果您不准备在发布新文章时每次准备发布的每期文章中都没有更新所有链接,则不要在每篇文章的顶部都包含目录(TOC)。 如果将这些交叉发布到公司的公司博客中,则工作量是原来的2倍🙂
  2. 随着时间的流逝,您可能会得出自己宁愿偏离主要用例的结论,即您最初使用的Big Coding Example。 我宁愿展示更多的FP概念(例如,使用FP技术时的生硬,记忆,懒惰以及不同的心态),但我不能很好地适应以前完成的重构和我在一开始建立的TOC 。 如果您正在撰写有关特定概念的文章,通常会找到一个适当的示例来帮助说明手头的特定概念,并且仍然与读者相关。 随着时间的流逝,我将获得更好的洞察力,从而可以确定接下来要写的更好的东西以及要使用的更合适的示例。 下次,我将不得不寻找一种方法来给(更好:允许)我自己一些创作上的自由😉

阅读更多

  • 《功能性思维:语法惊人的范式》,尼尔·福特(Neil Ford)着,它展示了FP思维的新方式,并且以不同的方式处理问题。
  • 40分钟内的函数式编程Russ Olsen的Youtube视频解释说:“这些数学家用379页证明1 + 1 = 2。 让我们看看我们可以从中窃取什么好主意。”
  • 为什么不对函数进行规范编程? 理查德·费尔德曼(Richard Feldman)的Youtube视频,他解释了为什么OOP变得非常流行,以及FP为何不是常态。 正如您所知,他是Elm核心团队的成员,与FP有一定的联系。
  • (耦合)控制的倒置有关“托管功能”的深思熟虑的文章。 您想要抽象吗?

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

编程愉快! 🙂

翻译自: https://www.javacodegeeks.com/2019/12/functional-java-by-example-part-8-more-pure-functions.html

接口文档示例

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值