java 示例

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

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

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

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

这些都是这些部分:

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

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

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

快速了解异常

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

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

  • Java异常 (JavaWorld)
  • Java例外– GeeksforGeeks (geeksforgeeks.org)
  • 9种处理Java异常的最佳实践 (stackify.com)
  • 异常处理的最佳实践 (onjava.com)
  • Java异常面试问答 (journaldev.com)
  • 带有示例的Java中的异常处理 (beginnersbook.com)
  • Java异常处理(Try-catch) (hackerrank.com)
  • 十大Java异常处理最佳实践– HowToDoInJava (howtodoinjava.com)
  • Java中的异常处理和断言– NTU (ntu.edu.sg)
  • 异常处理:最佳实践指南 (dzone.com)
  • 在Java中处理异常的9种最佳实践 (dzone.com)
  • 修复7个常见Java异常处理错误 (dzone.com)
  • Java惯例->检查与未检查异常 (javapractices.com)
  • Java中异常的常见错误 MikaelStåldal的技术博客 (staldal.nu)
  • Java开发人员在使用异常时犯的11个错误 (medium.com/@rafacdelnero)
  • 检查异常是好是坏? (JavaWorld)
  • 检查异常:Java的最大错误 精简Java (literatejava.com)
  • 未经检查的异常-争议 (docs.oracle.com)
  • 已检查的异常问题 (artima.com)
  • Java例外:您(可能)做错了 (dzone.com)
  • Java理论与实践:异常辩论– IBM (ibm.com)
  • Java的检查异常是一个错误(这是我想做的 (radio-weblogs.com)
  • Buggy Java代码:Java开发人员最常犯的10个错误 Toptal (toptal.com)

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

  • Java输入流的错误处理– Javamex (javamex.com)
  • 在Java流中处理检查的异常 (oreilly.com)
  • JDK 8流中的异常处理 (azul.com)
  • 带有异常的Java 8功能接口 (slieb.org)
  • 重新打包流中的异常– blog @ CodeFX (blog.codefx.org)
  • 如何在Java 8 Stream中处理异常? –堆栈溢出 (stackoverflow.com)
  • 检查异常和流| Benji的博客 (benjiweber.co.uk)
  • 一个关于检查异常和Java 8 Lambda表达式的 故事 (javadevguy.wordpress.com)– 很好的战争故事!
  • hgwood / java8-streams-and-exceptions (github.com)

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

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

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

副作用

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

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

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

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

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

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

class FeedHandler {Webservice webserviceDocumentDb documentDbvoid 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.iddoc.status = 'processed'documentDb.update(doc)}private void updateToFailed(doc, e) {doc.status = 'failed'doc.error = e.messagedocumentDb.update(doc)}}

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

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

在上面的代码中catch (e)是Groovy catch (Exception e)简写catch (Exception e)

是的,这就是我们正在捕获的通用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 webserviceDocumentDb documentDbvoid 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.iddoc.status = 'processed'documentDb.update(doc)}private void updateToFailed(doc, e) {doc.status = 'failed'doc.error = e.messagedocumentDb.update(doc)}}

-

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

java 示例

java 示例_功能Java示例 第3部分–不要使用异常来控制流程相关推荐

  1. java 函数式编程 示例_功能Java示例 第8部分–更多纯函数

    java 函数式编程 示例 这是第8部分,该系列的最后一部分称为"示例功能Java". 我在本系列的每个部分中开发的示例是某种"提要处理程序",用于处理文档. ...

  2. 大数据 java 代码示例_功能Java示例 第7部分–将失败也视为数据

    大数据 java 代码示例 这是称为" Functional Java by Example"的系列文章的第7部分. 我在本系列的每个部分中开发的示例是某种"提要处理程序 ...

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

    java 示例 这是称为" Functional Java by Example"的系列文章的第5部分. 在上一部分中,我们停止了对文档的变异,并返回了数据的副本. 现在,我们需要 ...

  4. java 示例_功能Java示例 第4部分–首选不变性

    java 示例 这是称为" Functional Java by Example"的系列文章的第4部分. 在上一部分中,我们讨论了一些副作用,并且我想进一步详细说明如何通过将不可变 ...

  5. java 示例_功能Java示例 第2部分–讲故事

    java 示例 这是称为" Functional Java by Example"的系列文章的第2部分. 我在本系列的每个部分中开发的示例是某种"提要处理程序" ...

  6. java 函数式编程 示例_功能Java示例 第1部分–从命令式到声明式

    java 函数式编程 示例 功能编程(FP)的目的是避免重新分配变量,避免可变的数据结构,避免状态并全程支持函数. 如果将功能性技术应用于日常Java代码,我们可以从FP中学到什么? 在这个名为&qu ...

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

    对象作为参数示例java 这是称为" Functional Java by Example"的系列文章的第6部分. 我在本系列的每个部分中开发的示例是某种"提要处理程序& ...

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

    这是称为" Functional Java by Example"的系列文章的第3部分. 我在本系列的每个部分中发展的示例是某种"提要处理程序",用于处理文档. ...

  9. java 示例_最佳Java示例

    java 示例 什么是Java? (What is Java?) Java is a programming language developed by Sun Microsystems in 199 ...

最新文章

  1. Java5线程并发库之保障变量的原子性操作
  2. iOS - UISearchController
  3. NUMBER_GET_NEXT 获取编号年度问题
  4. fat32硬盘格式能在苹果系统里读写操作吗?
  5. JavaScript: The Good Parts 学习随笔(三)
  6. Spark DataFrames DataSet
  7. android 蓝牙播放器,Android 蓝牙音乐播放器
  8. 三维实时云渲染平台解决方案
  9. 支付宝二维码可以抓包更改金额_支付宝到账铃声,金额可以随意改。
  10. Android 虚拟机EditText键盘无法输入解决方法
  11. [原创]网上一位叫啊松得网友提供,本人搜集!!
  12. 什么是云计算数据中心互连,云计算数据中心互连是如何运作的
  13. java正则表达式 后顾_正则表达式 - CrazyManDF的个人空间 - OSCHINA - 中文开源技术交流社区...
  14. 计算机控制系统的输入输出信号,工业控制系统的输入与输出信号
  15. 小蚁摄像机存储到计算机,小蚁摄像机电脑客户端
  16. AirDisk产品A6/A4支持OTG数据线连接到手机吗
  17. 中国使用互联网每年向美国付5000亿超国防预算数额
  18. Java集合篇:Map接口、Map接口的实现类、Collections集合工具类
  19. Algebraic Teamwork
  20. java gui 跳转_java GUI点击一个按钮怎么跳转到下一个窗口

热门文章

  1. [AtCoder Regular Contest 123] 题解
  2. CF1628A-Meximum Array【二分】
  3. P4343-[SHOI2015]自动刷题机【二分答案】
  4. Ch4302-IntervalGCD【线段树,树状数组,GCD】
  5. codeforces229 D. Towers(dp+贪心)
  6. 【数论】数表(P3312)
  7. [XSY3381] 踢罐子(几何)
  8. 4、java中的流程控制(程序结构)
  9. Sentinel(十七)之启动配置项
  10. 一个简单的例子,学习自定义注解和AOP