功能Java示例 第3部分–不要使用异常来控制流程

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

我在本系列的每个部分中发展的示例是某种“提要处理程序”,用于处理文档。 在前面的部分中,我从一些原始代码开始,并应用了一些重构来描述“什么”而不是“如何”。

为了帮助代码向前发展,我们需要摆脱良好的java.lang.Exception (免责声明:我们实际上无法摆脱它)这就是其中的内容。

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

这些都是这些部分:

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

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

快速了解异常

自Java 1.0以来,我们的java.lang.Exception一直存在-基本上在好时机和其他时候成为我们的敌人。

关于它们的讨论不多,但是如果您想阅读一些资源,这是我的最爱:

您已经在Java 8上了吗? 生活变得好多了! 我… 呃…哦,等等。

好的,看来您不可能正确地做到这一点。

至少,阅读上述名单后,我们现在完全取决于对速度的话题��

幸运的是,我不必再写一篇博客文章,介绍上面的文章已经涵盖了95%的内容,但是在这里,我将重点讨论代码中实际存在的一个Exception ��

副作用

既然您正在阅读这篇文章,您可能会对为什么这一切都与函数式编程有关感兴趣。

在以更“实用的方式”处理代码的过程中,您可能会遇到“副作用”一词,这是一件“坏事”。

在现实世界中, 副作用是您不希望发生的事情 ,您可能会说它等同于“例外”情况(您会例外说明),但是在函数式编程上下文中它具有更严格的含义。

维基百科有关副作用的文章说:

副作用(计算机科学)在计算机科学中,如果函数或表达式在返回范围之外修改了某些状态或与其调用函数或外界有可观察的交互作用,则称该函数或表达式具有副作用。 …在函数式编程中,很少使用副作用。

因此,在本系列的前两篇文章之后,让我们看看我们的FeedHandler代码当前的样子:

class FeedHandler {

  Webservice webservice
  DocumentDb documentDb

  void handle(List<Doc> changes) {

    changes
      .findAll { doc -> isImportant(doc) }
      .each { doc ->

      try {
        def resource = createResource(doc)
        updateToProcessed(doc, resource)
      } catch (e) {
        updateToFailed(doc, e)
      }
    }
  }

  private Resource createResource(doc) {
    webservice.create(doc)
  }

  private boolean isImportant(doc) {
    doc.type == 'important'
  }

  private void updateToProcessed(doc, resource) {
    doc.apiId = resource.id
    doc.status = 'processed'
    documentDb.update(doc)
  }

  private void updateToFailed(doc, e) {
    doc.status = 'failed'
    doc.error = e.message
    documentDb.update(doc)
  }

}

在一个地方,我们尝试捕获异常,在那儿,我们循环浏览重要的文档,并尝试为其创建“资源”(无论是什么)。

try {
  def resource = createResource(doc)
  updateToProcessed(doc, resource)
} catch (e) {
  updateToFailed(doc, e)
}

在上面的代码中catch (e)catch (Exception e) Groovy缩写。

是的,这就是我们正在捕获的通用java.lang.Exception 。 可以是任何例外,包括NPE。

如果createResource方法没有引发异常,则将文档(“ doc”)更新为“已处理”,否则将其更新为“失败”。 顺便说一句,即使updateToProcessed也会引发异常,但是对于当前的讨论,我实际上只对成功创建资源感兴趣。

因此,上面的代码可以工作 (我已经通过单元测试来证明它:-)),但是我对try-catch语句不满意。 我只对成功创建资源感兴趣,而且很傻,我只能提出createResource要么返回成功的资源, 要么抛出异常。

抛出异常以表示出了点问题,躲开闪避,让调用者捕获该异常以进行处理,这是为什么发明了异常的原因呢? 而且比返回null更好吗?

它一直在发生。 采取一些我们喜欢的框架,例如JPA规范中的 EntityManager#find

啊! 返回null

返回值:
找到的实体实例;如果该实体不存在,则返回null

错误的例子。

函数式编程鼓励使用无副作用的方法(或:函数),以使代码更易于理解且更易于推理。 如果某个方法仅接受某些输入并每次都返回相同的输出(这使其成为一个函数),则各种优化都可以在后台进行,例如通过编译器或缓存,并行化等。

我们可以再次用函数(计算出的值)替换函数,这称为参考透明度

在上一篇文章中,我们已经将一些逻辑提取到了自己的方法中,例如下面的isImportant 。 给定相同的文档(具有相同的 type属性)作为输入,我们每次都会获得相同的 (布尔值)输出。

boolean isImportant(doc) {
  doc.type == 'important'
}

这里没有可观察到的副作用,没有全局变量被突变,没有日志文件被更新–它只是塞进,塞出

因此,我要说的是,通过我们的传统异常与外界交互的函数很少在函数式编程中使用。

我想做得更好更好

可选救援

正如石磊韦伯表示它:

关于如何在Java中有效使用异常有不同的观点。 有些人喜欢检查异常,有些人则认为这是一次失败的实验,他们更喜欢独占使用未检查异常。 其他人则完全避开异常,而赞成传递和返回诸如Optional或Maybe之类的类型。

好的,让我们尝试一下Java 8的Optional以便发出是否可以创建资源的信号。

让我们更改我们的webservice接口和createResource方法,以在Optional包装并返回我们的资源:

//private Resource createResource(doc) {
private Optional<Resource> createResource(doc) {
  webservice.create(doc)
}

让我们更改原始的try-catch

try {
  def resource = createResource(doc)
  updateToProcessed(doc, resource)
} catch (e) {
  updateToFailed(doc, e)
}

map (处理资源)和orElseGet (处理空的可选):

createResource(doc)
  .map { resource ->
    updateToProcessed(doc, resource)
  }
  .orElseGet { /* e -> */
    updateToFailed(doc, e)
  }

很棒的createResource方法:返回正确结果,或者为空结果。

等一下! 唯一的例外e我们需要传递到updateToFailed走了 :我们有一个空的Optional替代。 我们不能存储的原因失败的原因 -这是我们做的必要性。

可能是Optional只是表示“缺席”,并且是我们此处目的不正确的工具。

出色的完成

如果没有try-catchmap-orElseGet ,我确实喜欢代码开始更多地反映操作“流程”的方式。 不幸的是,使用Optional更适合“得到一些东西”或“什么也没有得到”(这也建议使用maporElseGet类的名称),并且没有给我们提供记录失败原因的机会。

还有什么方法能够获得成功的结果或失败的原因,而仍然接近我们的阅读方式呢?

Future 。 更好的是: CompletableFuture

CompletableFuture (CF)知道如何返回值,这类似于Optional 。 通常,CF用于将来获取值 ,但这不是我们想要将其用于…的原因。

Javadoc

……支持……在完成时触发的行动的未来。

跳动,它可以表示“异常”完成 -给我机会对此采取行动。

让我们更改maporElseGet

createResource(doc)
  .map { resource ->
    updateToProcessed(doc, resource)
  }
  .orElseGet { /* e -> */
    updateToFailed(doc, e)
  }

thenAccept (处理成功)并exceptionally (处理失败):

createResource(doc)
  .thenAccept { resource ->
    updateToProcessed(doc, resource)
  }
  .exceptionally { e ->
    updateToFailed(doc, e)
  }

CompletableFuture#exceptionally方法接受一个带有我们实际失败原因的异常e的函数。

您可能会想: tomayto,tomahto。 首先,我们进行了try-catch ,现在我们进行了thenAccept-exceptionally ,那么有什么大不同?

好吧,我们显然不能摆脱特殊情况,但我们现在正在像Functionalville的居民那样思考:我们的方法开始成为函数 ,告诉我们有什么东西有事。

考虑到这是我们在第4部分中需要进行的少量重构,而在第5部分中,甚至更多地限制了代码中的副作用。

现在就这样

作为参考,这是重构代码的完整版本。

class FeedHandler {

  Webservice webservice
  DocumentDb documentDb

  void handle(List<Doc> changes) {

    changes
      .findAll { doc -> isImportant(doc) }
      .each { doc ->
        createResource(doc)
        .thenAccept { resource ->
          updateToProcessed(doc, resource)
        }
        .exceptionally { e ->
          updateToFailed(doc, e)
        }
      }
  }

  private CompletableFuture<Resource> createResource(doc) {
    webservice.create(doc)
  }

  private boolean isImportant(doc) {
    doc.type == 'important'
  }

  private void updateToProcessed(doc, resource) {
    doc.apiId = resource.id
    doc.status = 'processed'
    documentDb.update(doc)
  }

  private void updateToFailed(doc, e) {
    doc.status = 'failed'
    doc.error = e.message
    documentDb.update(doc)
  }

}

-

翻译自: https://www.javacodegeeks.com/2018/01/functional-java-example-part-3-dont-use-exceptions-control-flow.html

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值