dubbo全局异常处理_dubbo和spring mvc全局异常处理器实现

本文详细介绍了如何在Spring MVC中使用@ControllerAdvice和@ExceptionHandler实现全局异常处理,并探讨了由于Spring的全局异常处理对Dubbo调用不生效的问题。通过实现Filter并结合Dubbo的拦截扩展,可以在服务调用时捕获并处理异常,保持处理方式的一致性。文中还给出了具体的代码示例和异常处理流程。
摘要由CSDN通过智能技术生成

spring mvc 的全局异常处理器

@ExceptionHandler

当它在一个 controller 内部声明时,它将被用于那个controller(或它的子类)的 @RequestMapping方法抛出的异常. 你也可以在 @ControllerAdvice 类里面声明 @ExceptionHandler 方法,它将会处理很多controller的 @RequestMapping方法抛出的异常. 我的例子就是写一个异常处理类,其被@ControllerAdvice注解。

例如:

@ControllerAdvice

public class BizExceptionFilter implements Filter {

@ExceptionHandler(value = Exception.class)

public Object handleIOException(Exception e){

log.info("Catch exception", e);

String exceptionName=e.getClass().getName();

ResultExceptionInfoBean resultExceptionInfoBean=getInfoBean(exceptionName);

FResponse rsp = new FResponse();

rsp.setCode(resultExceptionInfoBean.getCode());

rsp.setMessage(resultExceptionInfoBean.getMessage());

rsp.setData(e);

return rsp;

}

@ExceptionHandler 的value可以设置一个需要被处理的异常数组. 如果一个异常被抛出并且包含在这个异常列表中, 然后就会调用 @ExceptionHandler 方法. 如果没有设置value,

那么就会使用参数里面的异常.

和标准controller的 @RequestMapping 方法很相似, @ExceptionHandler 方法的参数值和返回值相当灵活. 比如说, HttpServletRequest 可以在 Servlet 环境中被接收, PortletRequest 在 Portlet 环境中被接收. 返回值可以是 String, 它将解释为一个视图, 可以是 ModelAndView 对象, 可以是 ResponseEntity 对象, 或者你可以添加 @ResponseBody 方法直接返回消息.

dubbo的异常处理

由于spring 的全局异常处理只能对http请求有效,所以对于dubbo的调用不起作用。

那么为了实现对dubbo调用异常的处理,我们可以使用dubbo的拦截扩展.

具体配置方式详见 API.

将扩展方法写到spring 的全局异常处理器中,使其继承Filter,然后配置服务方调用拦截。

当服务被调用时就会执行该方法。

例如:

@ControllerAdvice

public class BizExceptionFilter implements Filter {

@Override

public Result invoke(Invoker> invoker, Invocation invocation) throws RpcException {

log.debug("BizExceptionFilter:{},{}", invoker.getInterface(),

JsonUtil.jsonFromObject(invocation.getArguments()));

Result result = invoker.invoke(invocation);

Object realResult = result.getValue();

FResponse rsp = new FResponse();

if (result.hasException()) {

try {

ExceptionHandlerMethodResolver resolver=new ExceptionHandlerMethodResolver(this.getClass());

Exception exception=(Exception) result.getException();

Method method=resolver.resolveMethod(exception);

realResult = method.invoke(this, exception);

return new RpcResult(realResult);

} catch (Throwable e) {

log.error("Exception handler error. Caused Exception:{}", result.getException());

}

}

rsp.setCode("0000");

rsp.setMessage("Success");

rsp.setData(realResult);

return new RpcResult(rsp);

}

@ExceptionHandler(value = NullPointerException.class)

public Object handleIOException(NullPointerException e){

log.info("Catch exception", e);

String exceptionName=e.getClass().getName();

ResultExceptionInfoBean resultExceptionInfoBean=getInfoBean(exceptionName);

FResponse rsp = new FResponse();

rsp.setCode(resultExceptionInfoBean.getCode());

rsp.setMessage(resultExceptionInfoBean.getMessage());

rsp.setData(e);

return rsp;

}

@ExceptionHandler(value = IndexOutOfBoundsException.class)

public Object handleIOException(IndexOutOfBoundsException e){

log.info("Catch exception", e);

String exceptionName=e.getClass().getName();

ResultExceptionInfoBean resultExceptionInfoBean=getInfoBean(exceptionName);

FResponse rsp = new FResponse();

rsp.setCode(resultExceptionInfoBean.getCode());

rsp.setMessage(resultExceptionInfoBean.getMessage());

rsp.setData(e);

return rsp;

}

解析:

Result result = invoker.invoke(invocation);这行代码之前的代码,是在服务被调用钱执行,之后的代码是在服务被调用后执行。

Object realResult = result.getValue();获得执行结果.

result.hasException()是否抛出了异常.

A912450C-D6D7-4E83-85E4-12034B2569B4.png这段代码就是利用反射机制,获得当前的类的方法,然后调用该对象的有exception参数的方法,并将结果返回。

2017-3-15 更新

Exception exception=(Exception) result.getException(); 这行代码的作用是获得此类中被@ExceptionHandler注解的方法。

Method method=resolver.resolveMethod(exception);这行代码的作用是从被@ExceptionHandler注解的所有方法中找出value为exception的方法对象。

realResult = method.invoke(this, exception); 执行这个方法。

如图:

6D14283E-035B-443E-9FDA-276DA93643AB.png

这样保持了和spring 的全局异常处理一致,即被@ExceptionHandler注解的方法去处理异常。

spring 相关源码如下:

Paste_Image.png

测试:

我在服务类中直接抛出了异常。

8EB375E9-8266-4A82-BD85-0FD9BFEC085F.png

顺利执行:

CB3B887D-FB1D-432B-A9CA-150FE7E3E93C.png

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
@ControllerAdvice 是用于全局异常处理的注解,其作用是在 Spring MVC 中统一处理异常,可以减少重复代码的编写。但是在 Dubbo 中,由于 Dubbo 是一个 RPC 框架,不是基于 HTTP 协议的 Web 框架,所以 @ControllerAdvice 并不会生效。 在 Dubbo 中,如果需要实现全局异常处理,可以使用 Dubbo 提供的 Filter。Dubbo Filter 是一个类似于 Servlet Filter 的组件,可以在服务提供者和服务消费者之间进行拦截,实现一些通用的功能,例如:日志记录、安全验证、异常处理等。 具体来说,可以实现一个实现了 Filter 接口的类,并在其中重写 onInvoke() 方法。在该方法中,可以捕获到服务调用时抛出的异常,并进行处理。例如,可以将异常信息记录到日志中,或者返回一个自定义的错误信息。 示例代码如下: ```java public class ExceptionFilter implements Filter { @Override public Result invoke(Invoker<?> invoker, Invocation invocation) throws RpcException { try { // 调用服务 return invoker.invoke(invocation); } catch (Exception e) { // 异常处理 // 记录日志或者返回自定义的错误信息 return new RpcResult("调用服务出现异常"); } } } ``` 然后在 Dubbo 的配置文件中,将该 Filter 配置到服务提供者和服务消费者上即可: ```xml <dubbo:provider filter="exceptionFilter" /> <dubbo:consumer filter="exceptionFilter" /> ``` 注意,需要将上述代码中的 exceptionFilter 替换成实际的 Filter 类名。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值