优雅地在Java 8中处理异常的方法详解

 更新时间:2019年01月10日 09:08:59   作者:_pangzi  
异常处理是我们在日常开发中经常会遇到的一种需求,下面这篇文章主要给大家介绍了关于如何优雅地在Java 8中处理异常的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们来一起看看吧

前言

Java 8 引入的流 (Stream) API 和 lambda 表达式为我们打开了新世界的大门,自此之后我们也可以在 Java 中进行函数式编程了。然而,在实际工作中,许多小伙伴并不知道如何正确的在 lambda 中处理异常,今天就来给大家讲解一下。

小编给大家推荐一个Java技术交流群:937053620!群内提供设计模式、spring/mybatis源码分析、高并发与分布式、微服务、性能优化,面试题整合文档等免费资料!给大家提供一个交流学习的平台!

我们都知道,Java 异常分为检查异常和非检查异常。检查异常就是编译器要求开发者必须处理的异常,而非检查异常则没有这个要求。所以当我们需要调用某个抛出检查异常的方法时,必须明确捕获它:

myList.stream()
 .map(item ->
  try{
  return doSomething(item);
  } catch(MyException e){
  throw new RuntimeException (e);
  }
 })
 .forEach(System.out::printion);

如上面代码所示,我们捕获了 MyException 这个检查异常,然后将其转化为 RuntimeException 非检查异常,重新抛出。但是你自己心里面其实清楚的很,这不是最好的处理方式。

优化一: 提升可读性

如下所示,我们将方法体单独提取到 trySomething 方法中,这样的话,我们就可以使用一行代码完成 lambda 表达式,整个代码可读性也会提升不少:

myList.stream()
 .map(this::trySomething)
 .forEach(System.out::printion);

private Item trySomething(Item item) {
 try{
  return doSomething(item);
 } catch(MyException e){
  throw new RuntimeException (e);
 }
}

优化二: 复用代码

现在你已经解决了上述的问题,然而当我们再碰到需要处理异常的其它方法时,难道我们都要用 try ... catch ... 包裹一层吗?那样的话,你可以想象代码中可能到处都是这种类似的写法。为了避免陷入到这种重复的写法中,我们应该将上述代码片段抽象为一个小的工具类,专门用来干这件事情。你只需要定义一次,然后再需要的地方多次调用它就可以了。

为了实现这个目标,我们首先需要自己定义一个函数式接口,这个接口可能会抛出一个异常:

然后,我们来写一个静态帮助函数 wrap ,该方法接受一个函数式接口参数,在方法体内捕获检查异常,并抛出非检查异常 RuntimeException:

借助于 wrap 静态函数,现在你可以在 lambda 表达式中这么写了

优化三: 出现异常时继续运行

上述代码的可读性、抽象性已经很好了,然而还存在一个比较大的问题,那就是当出现异常的时候,你的 stream 代码会立即停止,不会接着处理下一个元素。大多数情况下,当抛出异常的时候,我们可能还想让 stream 继续运行下去。

我们与其抛出异常,将异常当成一种特殊的情况处理,还不如直接将异常当成是一个 “正常” 的返回值。即这个函数要么返回一个正确的结果,要么返回一个异常,所以我们现在需要定义一个新的封装类 Either,用来存储这两种结果。为了方便,我们将异常存储到 left 这个字段中,将正常返回的值存储到 right 这个字段中。下面就是 Either 类的一个简单示例:

public class Eithercl<L,R>{

  private final L Left:
  private final R right;

  private Either(L left, R right){
   this left=left;
   this right =right;
}
public static <L, R> Either,<L,R> Left( L value) {
 return new Either(value, null):
}
public static <L, R> Either<L, R> Right( R value) {
 return new Either(null, value)
}
public Optional<L> getleft() {
 return Optional. ofnullable(left)
}
public Optional<R> getright() {
 return Optional.ofnullable(right);
}
public boolean isleft() {
 return left I- null;
}
public boolean isright(){
 return right != null;
}
public < T> optional<T> mapleft(Function<? super L, T> mapper){
 if (isleft()) {
  return Optional of(mapper. apply(left));
  }
  return Optional empty();
}
public <T> Optional<T> mapright(Function<? super R, T> mapper) {
 if (isright()) {
   return Optional of(mapper. apply(right));
 }
 return Optionalempty();
}
public String tostring(){
 if (isleft()){
  return"Left(”+left+")";
 }
 return "Right("+ right +")";
 } 
}

现在我们需要再定义一个 lift 函数,该函数内部将 function 函数正常返回的值或者抛出的异常都使用 Either 类进行了一层封装

现在我们的代码变成这个样子了,也不用担心方法抛出异常会提前终止 Stream 了

优化四: 保留原始值

现在思考一个问题,如果在上述处理过程中,当结果是异常信息的时候,我们想要重试,即重新调用这个方法怎么办? 你会发现我们 Either 封装类没有保存最原始的这个值,我们丢掉了原始值,因此我们可以进一步优化,将原始值 t 也封装进 left 字段中,就像下面这样:

Pair 类是一个非常简单的封装类,用以封装两个值:

public class Pair<F, S> {
 public final F fst;
 public final S snd;

 private Pair(F fst, S snd){
   this fst fst;
   this snd= snd;
 }
public static <F, S> Pair<F, S> of(F fst, S snd){
 return new Pair<>(fst, snd);
 }
}

这样,当我们遇见异常的时候,我们可以从 Pair 中取出最原始的值 t,无论是想重试,还是做一些其他操作,都很方便了。

总结

我们经过上文一点一点地优化代码,得到了一个比较满意的在 Java 8 中处理异常的通用方式。其实,大家还可以关注 Github 上的有关函数式编程方面的库,比如 Javaslang ,它实现了多种多样的函数式帮助方法和封装类来帮助开发者写好 lambda 表达式。但是,如果你只是为了处理异常,而引入这么大的一个第三方库的话,就不太建议了哦~

好了,以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对脚本之家的支持。

相关文章

  • 关于Java从本地文件复制到网络文件上传

    关于Java从本地文件复制到网络文件上传

    这篇文章主要介绍了关于Java从本地文件复制到网络文件上传,File 和 IO 流其实是很相似的,都是将文件从一个地方转移到另一个地方,这也是流的特点之一,需要的朋友可以参考下
    2023-04-04
  • Java多线程Condition接口原理介绍

    Java多线程Condition接口原理介绍

    这篇文章主要介绍了Java多线程Condition接口原理介绍,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-09-09
  • 关于JVM垃圾回收的java.lang.ref.Finalizer问题

    关于JVM垃圾回收的java.lang.ref.Finalizer问题

    这篇文章主要介绍了关于JVM垃圾回收的java.lang.ref.Finalizer问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-05-05
  • ThreadLocal导致JVM内存泄漏原因探究

    ThreadLocal导致JVM内存泄漏原因探究

    ThreadLocal是JDK提供的线程本地变量机制,但若使用不当可能导致内存泄漏。正确的使用方式是在使用完后及时remove,或者使用弱引用等手段避免强引用导致的内存泄漏。在多线程编程中,合理使用ThreadLocal可以提高并发性能,但也需要注意其潜在的内存泄漏问题
    2023-04-04
  • Java日期工具类操作字符串Date和LocalDate互转

    Java日期工具类操作字符串Date和LocalDate互转

    这篇文章主要介绍了Java日期工具类操作字符串Date和LocalDate互转,文章首先通过需要先引入坐标展开主题的相关内容介绍,需要的朋友可以参一下
    2022-06-06
  • Spring Cloud + Nacos + Seata整合过程(分布式事务解决方案)

    Spring Cloud + Nacos + Seata整合过程(分布式事务解决方案)

    Seata 是一款开源的分布式事务解决方案,致力于在微服务架构下提供高性能和简单易用的分布式事务服务,这篇文章主要介绍了Spring Cloud + Nacos + Seata整合过程(分布式事务解决方案),需要的朋友可以参考下
    2022-03-03
  • mybatis接口绑定失效的解决

    mybatis接口绑定失效的解决

    这篇文章主要介绍了mybatis接口绑定失效的解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-11-11
  • java实现微信公众号扫一扫

    java实现微信公众号扫一扫

    这篇文章主要为大家详细介绍了java实现微信公众号扫一扫,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-04-04
  • Java 自定义Spring框架与核心功能详解

    Java 自定义Spring框架与核心功能详解

    Spring框架是由于软件开发的复杂性而创建的。Spring使用的是基本的JavaBean来完成以前只可能由EJB完成的事情。然而,Spring的用途不仅仅限于服务器端的开发
    2021-10-10
  • Spring Boot与Docker部署详解

    Spring Boot与Docker部署详解

    本篇文章主要介绍了Spring Boot与Docker部署详解,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-08-08

最新评论