WebFlux异常处理:onErrorReturn和onErrorResume

1 缘起

最近在学习WebFlux,
处理异常时遇到些问题,比如,Java直接抛出的异常无法直接被onErrorReturn和onErrorResume捕获,
但是,在map或者flatMap等方法之后的异常又可以直接被捕获,
于是,进行了测试,发现,onErrorReturn和onErrorResume可以捕获的异常是Throwable类型或者Mono或Flux包装的类型。
当我们在使用WebFlux中的onErrorReturn和onErrorResume捕获异常时,有两种方式:
(1)当原生Java运行时异常抛出时,需要使用Mono或Flux包装:Mono.error或FluxError;
(2)在map或者flatMap等其他原生方法中抛出的异常可以直接被捕获,无需包装。
特分享如下,帮助读者轻松应对知识交流与考核。
在这里插入图片描述

2 异常处理

异常处理是软件开发中非常重要的一环,
在业务系统中,我们需要捕获异常,给出合适的描述信息,让调用者清晰这是哪一个环节的异常。
在计算相关的业务中,通常对计算结果有更多的要求,比如,程序内部出现异常后,不能影响计算结果的输出,需要有一个保底的计算结果,比如在广告推荐系统中的广告出价,即使程序内部出现异常,仍要返回最终的广告出价,保证流程可以正常继续往下走。
当然,不同的业务系统有不同的需求,有的需要直接将异常暴露出去,有的需要记录但是给出保底的结果,但是,都要都要对异常进行处理。

2.1 WebFlux异常处理

WebFlux中的异常处理与SpringMVC中的异常处理存在不同的地方,
相同:都可使用全局异常拦截;
不同:WebFlux可以使用自定义的异常处理方法,如onErrorReturn和onErrorResume。
在WebFlux中,onErrorReturn和onErrorResume可以处理的异常是经过Mono或Flux包装的。
可以这样理解:Java直接抛出的运行时异常,onErrorReturn和onErrorResume无法直接捕获,需要通过Mono.error或者Flux.error包装,才能被正常捕获,当然,如果启用了全局异常捕获,则无需包装。
下面看一下onErrorReturn和onErrorResume的源码:

onErrorReturn

在这里插入图片描述

onErrorReusme

在这里插入图片描述

由源码可知,onErrorReturn中调用了onErrorResume,不同的是入参和返回值,onErrorReturn入参是固定值,onErrorResume入参是函数,可以自定义处理逻辑。onErrorResume的参数为Functions<? super Throwable, ? extends Mono>,由这个泛型参数可知,onErrorResume只能处理Throwable类型和Mono及其子类的类型数据,所以,无法直接捕获Java原生运行时异常。Java可抛出异常是Error和Exception的父类,因此,任何运行时异常都不能被onErrorResume或者onErrorReturn捕获,除非使用Mono.error或者Flux.error包装起来,才能正常捕获。
Java异常关系:https://blog.csdn.net/Xin_101/article/details/110210485

2.2 样例

前提条件:未开启全局异常捕获

  • 接口
@GetMapping("/mathematics/operation/flow/mono")
    @ApiOperation("连续流测试mono")
    public Mono<Response<Integer>> mathematicsOperationFlowUnderMono(@RequestParam Integer var1, @RequestParam Integer var2) {

        return mathematicsOperationService.divideMono(var1, var2)
                .onErrorReturn(10)
                .map(addResult -> {
                    logger.info(">>>>>>Add result:{}", addResult);
                    return Response.success(addResult);
                })
                .onErrorResume(ex -> {
                    logger.info(">>>>>>Error resume:", ex);
                    return Mono.just(Response.fail(-100));
                });
    }

2.2.1 原生运行时异常

Java原生异常即Throwable子类的错误和异常:Error和Exception。
以运行时异常为例,当程序出现异常时,接口返回的结果是WebFlux的信息,而不是开发者自定义的响应模板。

  • 异常处理
    @Override
    public Mono<Integer> divideMono(Integer var1, Integer var2) {
        try{
            return Mono.just(var1 / var2);
        } catch(Exception ex) {
            throw new RuntimeException("自定义运行时异常");
        }
    }

返回的信息如下:
在这里插入图片描述

2.2.2 Mono包装异常

onErrorReturn和onErrorResume可以处理Mono或Flux包装的异常,
包装如下:

  • 异常处理
    @Override
    public Mono<Integer> divideMono(Integer var1, Integer var2) {
        try{
            return Mono.just(var1 / var2);
        } catch(Exception ex) {
            return Mono.error(ex);
        }
    }

包装后的异常可以被哦那Error Return捕获,接口中在onErrorReturn中使用了固定数值:10,因此返回的结果是10,结果如下:
在这里插入图片描述

2.2.3 原生Mono异常

由上面可知,WebFlux可以捕获处理Mono或Flux包装的异常信息,
因此,当我们使用map或者flatMap处理数据时,可以直接使用onErrorReturn和onErrorResume捕获异常,map和flatMap中产生的异常会被直接捕获,无需进行包装。
为了测试,我们在map中添加了1/0,运行时会抛出异常,然后在onErrorResume中捕获异常,并返回-100。

    @GetMapping("/mathematics/operation/flow/mono")
    @ApiOperation("连续流测试mono")
    public Mono<Response<Integer>> mathematicsOperationFlowUnderMono(@RequestParam Integer var1, @RequestParam Integer var2) {

        return mathematicsOperationService.divideMono(var1, var2)
                .onErrorReturn(10)
                .onErrorResume(ex -> {
                    logger.info(">>>>>>Error resume -1:", ex);
                    return Mono.just(-1);})
                .map(addResult -> {
                    logger.info(">>>>>>Add result:{}", addResult);
                    int a = 1/0;
                    return Response.success(addResult);
                })
                .onErrorResume(ex -> {
                    logger.info(">>>>>>Error resume:", ex);
                    return Mono.just(Response.fail(-100));
                });
    }

当出现异常时,onErrorResume会直接捕获,并返回自定义的数据,如上面的-100,
结果如下图所示。

在这里插入图片描述

3 小结

(1)WebFlux处理异常可以使用onErrorReturn和onErrorResume;
(2)onErrorReturn和onErrorResume可以处理的异常是Throwable类型以及Mono或者Flux及其子类包装的异常;
(3)使用Mono.error和Flux.error包装异常或者在map,flatMap之后处理异常。

  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
WebFlux全局异常处理可以通过自定义全局异常处理器来实现。在WebFlux中,我们可以使用`@ControllerAdvice`注解来定义一个全局异常处理器类。下面是一个简单的示例: ```java @ControllerAdvice public class GlobalExceptionHandler { @ExceptionHandler(Exception.class) public ResponseEntity<ErrorResponse> handleException(Exception ex) { ErrorResponse errorResponse = new ErrorResponse(HttpStatus.INTERNAL_SERVER_ERROR, "An error occurred"); return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).body(errorResponse); } @ExceptionHandler(WebExchangeBindException.class) public ResponseEntity<ErrorResponse> handleBindException(WebExchangeBindException ex) { ErrorResponse errorResponse = new ErrorResponse(HttpStatus.BAD_REQUEST, "Invalid request"); return ResponseEntity.status(HttpStatus.BAD_REQUEST).body(errorResponse); } // 添加其他异常处理方法... // 自定义错误响应类 private static class ErrorResponse { private final HttpStatus status; private final String message; public ErrorResponse(HttpStatus status, String message) { this.status = status; this.message = message; } // getter 方法... } } ``` 在上述示例中,`@ControllerAdvice`注解用于声明一个全局异常处理器类。`@ExceptionHandler`注解用于指定处理特定异常的方法。在方法中,我们可以根据具体需求创建自定义的错误响应对象,并将其包装在`ResponseEntity`中返回给客户端。 注意,在WebFlux中,全局异常处理器处理的是异步请求,因此需要返回`Mono<ResponseEntity>`或`Flux<ResponseEntity>`类型的结果。 以上是一个简单的WebFlux全局异常处理的示例,你可以根据实际需求进行扩展和修改。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

天然玩家

坚持才能做到极致

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值