`
y806839048
  • 浏览: 1126734 次
  • 性别: Icon_minigender_1
  • 来自: 上海
文章分类
社区版块
存档分类
最新评论

dubbo异常处理

阅读更多

      dubbo有自己的异常处理机制,当服务端抛出一个dubbo可以处理传递的异常时,会直接在客户端上再次抛出,由开发者自己去处理。注意:这里说的不是所有异常,而是dubbo可以处理传递的异常,具体这个后边再说。

     先看两段代码,接口代码:

     

[java] view plain copy
 
  1. public interface IPersonService {  
  2.     public Person getPerson(long id);  
  3.     /** 
  4.      * 根据名称获取 
  5.      *  
  6.      * @param name 
  7.      * @return 
  8.      */  
  9.     public Person getByNick(String name);  
  10. }  

      简单实现:

 

    

[java] view plain copy
 
  1. public class PersonServiceImpl implements IPersonService {  
  2.     private final static AtomicInteger ID = new AtomicInteger();  
  3.   
  4.     @Override  
  5.     public Person getPerson(long id) {  
  6.         Person person = new Person();  
  7.         person.setId(id);  
  8.         person.setName("某某");  
  9.         try {  
  10.             Thread.sleep(2000l);  
  11.         } catch (InterruptedException e) {  
  12.             // TODO Auto-generated catch block  
  13.             e.printStackTrace();  
  14.         }  
  15.         return person;  
  16.     }  
  17.   
  18.     @Override  
  19.     public Person getByNick(String name) {  
  20.         name = name.trim();  
  21.         Person person = new Person();  
  22.         person.setId(ID.getAndIncrement());  
  23.         person.setName(name);  
  24.         return person;  
  25.     }  
  26.   
  27. }  

        这两段代码很简答,先看getByNick方法,根据用户名称获取用户信息,里面有一个去空格的操作(主要为了触发异常),正常调用是没有问题的,但如果传入null,就会抛出很常见且低级的空指针异常。我们看下调用代码:

 

   

[java] view plain copy
 
  1. ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext(  
  2.                 new String[] { "app-dubbo-consumer.xml" });  
  3.         context.start();  
  4.         IPersonService personService = context.getBean(IPersonService.class);  
  5.         // 正常输入参数  
  6.         outString(personService.getByNick("某某"));  
  7.         // 输入null  
  8.         outString(personService.getByNick(null));  

     运行后,首先会打印{"id":0,"name":"某某"},然后出现java.lang.NullPointerException,一切在我们的预料内,dubbo把服务端的空指针异常传递给客户端了。

 

     正常来说,空指针异常是不应该出现的,而且客户端遇到这个错误肯定直接懵了,所以我们做下简单的修改,服务端代码:

  

[java] view plain copy
 
  1. @Override  
  2.     public Person getByNick(String name) {  
  3.         if (name == null) {  
  4.             throw new RuntimeException("亲你咋没输入昵称呢");  
  5.         }  
  6.         name = name.trim();  
  7.         Person person = new Person();  
  8.         person.setId(ID.getAndIncrement());  
  9.         person.setName(name);  
  10.         return person;  
  11.     }  

         客户端再次调用结果:

 

  

[java] view plain copy
 
  1. Exception in thread "main" java.lang.RuntimeException: 亲你咋没输入昵称呢  
  2.     at org.dubbo.provider.PersonServiceImpl.getByNick(PersonServiceImpl.java:28)  
  3.     at com.alibaba.dubbo.common.bytecode.Wrapper1.invokeMethod(Wrapper1.java)  

     结果友好了多,甚至你可以直接对exception获取异常信息作为输出。

 

       上边提到过当服务端抛出一个dubbo可以处理传递的异常时,会直接在客户端上再次抛出,但不是所有的异常都是dubbo可以处理传递的,如下边的代码:

   

[java] view plain copy
 
  1. @Override  
  2.     public Person getByNick(String name) {  
  3.         if (name == null) {  
  4.             // 第三方异常  
  5.             throw new PersistenceException("mybatis插入异常");  
  6.         }  
  7.         name = name.trim();  
  8.         Person person = new Person();  
  9.         person.setId(ID.getAndIncrement());  
  10.         person.setName(name);  
  11.         return person;  
  12.     }  

 

        这里我们模拟抛出了一个mybatis的异常,在客户端调用会像上边的结果一样吗?答案是否定的,看下输出结果:

 

[java] view plain copy
 
  1. Exception in thread "main" java.lang.RuntimeException: org.apache.ibatis.exceptions.PersistenceException: mybatis插入异常  
  2. org.apache.ibatis.exceptions.PersistenceException: mybatis插入异常  
  3.     at org.dubbo.provider.PersonServiceImpl.getByNick(PersonServiceImpl.java:32)  
  4.     at com.alibaba.dubbo.common.bytecode.Wrapper1.invokeMethod(Wrapper1.java)  

       不要感觉奇怪,这个也是在可以接受的范围内,因为PersistenceException异常类在客户端是不存在的,所以不可能接收到PersistenceException异常,dubbo把他进行了封装。

 

        针对这点,在接口包中里面定义了一个全局的异常类,注意一定是接口所在的工程中,如:UicException(用户模块异常),这种方案也是官方建议的,服务端代码如下:

  

[java] view plain copy
 
  1. @Override  
  2.     public Person getByNick(String name) {  
  3.         if (name == null) {  
  4.             // 自定义异常  
  5.             throw new UicException("mybatis插入异常"0);  
  6.         }  
  7.         name = name.trim();  
  8.         Person person = new Person();  
  9.         person.setId(ID.getAndIncrement());  
  10.         person.setName(name);  
  11.         return person;  
  12.     }  

 

           错误信息如下

 

[java] view plain copy
 
  1. Exception in thread "main" org.dubbo.api.exception.UicException: mybatis插入异常  
  2.     at org.dubbo.provider.PersonServiceImpl.getByNick(PersonServiceImpl.java:30)  

         这个正是我们想要的异常信息,上边特别提到异常一定要在接口所在的工程中,如果异常类不在接口工程中,而是在另一个服务端和客户端都引入的包中呢?我们曾经碰到这样一个情况,有一个common的异常类放在一个很底层的工具包内,接口工程引入了这个包,在服务端抛出的异常都都是这个commonexception,一厢情愿的认为客户端会正常去捕获处理commonexception。

 

         但结果很意外,客户端出现的异常跟上边抛出的PersistenceException情况一样,dubbo用RuntimException进行了包装,我们无法从异常中获取有效的信息!遇到这种情况有点发懵,这个异常类在客户端和服务端都有呀,为啥不能正确接收呢。还好之前看dubbo源码的时候大概记得异常处理的位置,很好找到了目标代码:

 

[java] view plain copy
 
  1. if (result.hasException() && GenericService.class != invoker.getInterface()) {  
  2.                try {  
  3.                    Throwable exception = result.getException();  
  4.   
  5.                    // 如果是checked异常,直接抛出  
  6.                    if (! (exception instanceof RuntimeException) && (exception instanceof Exception)) {  
  7.                        return result;  
  8.                    }  
  9.                    // 在方法签名上有声明,直接抛出  
  10.                    try {  
  11.                        Method method = invoker.getInterface().getMethod(invocation.getMethodName(), invocation.getParameterTypes());  
  12.                        Class<?>[] exceptionClassses = method.getExceptionTypes();  
  13.                        for (Class<?> exceptionClass : exceptionClassses) {  
  14.                            if (exception.getClass().equals(exceptionClass)) {  
  15.                                return result;  
  16.                            }  
  17.                        }  
  18.                    } catch (NoSuchMethodException e) {  
  19.                        return result;  
  20.                    }  
  21.   
  22.                    // 未在方法签名上定义的异常,在服务器端打印ERROR日志  
  23.                    logger.error("Got unchecked and undeclared exception which called by " + RpcContext.getContext().getRemoteHost()  
  24.                            + ". service: " + invoker.getInterface().getName() + ", method: " + invocation.getMethodName()  
  25.                            + ", exception: " + exception.getClass().getName() + ": " + exception.getMessage(), exception);  
  26.   
  27.                    // 异常类和接口类在同一jar包里,直接抛出  
  28.                    String serviceFile = ReflectUtils.getCodeBase(invoker.getInterface());  
  29.                    String exceptionFile = ReflectUtils.getCodeBase(exception.getClass());  
  30.                    if (serviceFile == null || exceptionFile == null || serviceFile.equals(exceptionFile)){  
  31.                        return result;  
  32.                    }  
  33.                    // 是JDK自带的异常,直接抛出  
  34.                    String className = exception.getClass().getName();  
  35.                    if (className.startsWith("java.") || className.startsWith("javax.")) {  
  36.                        return result;  
  37.                    }  
  38.                    // 是Dubbo本身的异常,直接抛出  
  39.                    if (exception instanceof RpcException) {  
  40.                        return result;  
  41.                    }  
  42.   
  43.                    // 否则,包装成RuntimeException抛给客户端  
  44.                    return new RpcResult(new RuntimeException(StringUtils.toString(exception)));  
  45.                } catch (Throwable e) {  
  46.                    logger.warn("Fail to ExceptionFilter when called by " + RpcContext.getContext().getRemoteHost()  
  47.                            + ". service: " + invoker.getInterface().getName() + ", method: " + invocation.getMethodName()  
  48.                            + ", exception: " + e.getClass().getName() + ": " + e.getMessage(), e);  
  49.                    return result;  
  50.                }  
  51.            }  


          看完源码以后,做出了新的设计,CommonException不变,各个接口模块(maven工程为单位)单独定义异常对象继承CommonException(保证异常类和接口在同一模块,然后用里氏代换)每个模块抛出自己的模块异常(如用户模块抛出UicException),客户端中用CommonException统一捕获处理。

 

          这里还要定义两个拦截器,首先是服务端,保证所有抛出的异常是当前模块的异常,代码如下:

 

[java] view plain copy
 
  1. public class UICExceptionAspect {  
  2.     private static Log log = LogFactory.getLog(UICExceptionAspect.class);  
  3.   
  4.     public void afterThrowing(Exception ex) {  
  5.   
  6.         if (ex instanceof UICException) {  
  7.             throw (UICException) ex;  
  8.         } else if (ex instanceof DayimaException) {  
  9.             DayimaException de = (DayimaException) ex;  
  10.             throw new UICException(de.getErrCode(), de.getErrMsg());  
  11.         } else {  
  12.             log.error(ex.getMessage(), ex);  
  13.             throw new UICException("1", ex.getMessage());  
  14.         }  
  15.   
  16.     }  
  17. }  


         其次是客户端的,保证异常可以正确的友好的输出,所有CommonException可以直接输出(获取根据错误码获取错误信息),非CommonException异常根据自己需要去处理,如果是dubbo自带异常肯定要屏蔽异常信息,如打印日志后输出“网络异常”。

 

       还有另一种dubbo调用方案,普通service层外边嵌套一层用来做dubbo的服务,普通service层处理了事务之类,dubbo服务层每一个方法都是客户端要引用的,直接调用普通service层方法,但做了手动的try catch处理,封装自己的返回码,客户端只需要根据返回码去做处理,这种开发成本和文档成本有点高,没太深入去考虑。

        以上是我自己工作中的dubbo异常实践,以后会继续写些其他的心得,记录自己的成长。

分享到:
评论

相关推荐

    dubbo捕获自定义异常_dubbo异常捕获_dubbo异常_自定义异常_捕捉异常_

    在分布式服务框架 Dubbo 中,异常处理是必不可少的一部分。Dubbo 提供了强大的异常处理机制,使得服务提供者能够向消费者传递自定义异常,从而帮助消费者更好地理解和处理服务调用中的错误情况。本文将深入探讨如何...

    dubbo示例代码dubbo-sample

    Dubbo提供了多种失败处理策略,如重试、快速失败、忽略异常、回调恢复等。这有助于系统在服务异常时保持稳定。在`dubbo-samples-fault-tolerance`模块中,你可以看到这些策略的实现和应用场景。 7. **服务调用方式...

    incubator-dubbo-dubbo-2.6.1

    6. **服务监控**:Dubbo集成了服务调用统计和监控,可以实时查看服务的调用次数、调用时间、异常比例等信息,便于运维和调优。 7. **SPI扩展机制**:Dubbo提供了Service Provider Interface(SPI)机制,允许开发者...

    dubbo监控dubbo-monitor

    2. **异常报警**:当服务出现异常时,如超时、失败率上升等,Dubbo-Monitor可以通过配置发送报警通知,确保问题能得到及时处理。 3. **调用详情**:可以查看每个具体调用的详细信息,包括调用时间、调用耗时、参数...

    dubbo、dubbo-admin

    此外,还能查看服务的元数据、监控服务的调用情况、管理服务的版本以及处理服务的异常。 5. **监控与调试**: - Dubbo-admin提供了丰富的监控指标,如调用次数、成功率、平均耗时等,这对于诊断和优化服务性能非常...

    zookeeper+dubbo+spring

    然后,Dubbo会处理服务的调用,包括选择合适的服务器进行负载均衡,处理可能出现的网络异常,保证服务的高可用性。Spring在这个过程中起到了胶水的作用,将这些组件无缝地整合在一起,提供了统一的配置管理和依赖...

    nodejs使用原生的dubbo协议打通了dubbo的rpc方法调用.

    6. **错误处理和调试**:学习如何处理网络通信中可能出现的异常情况,如网络中断、超时、数据解析错误等,并进行有效的错误调试。 7. **性能优化**:考虑如何优化通信效率,比如减少网络传输的数据量,提高并发处理...

    dubbo-admin源码

    八、异常处理与日志记录 在源码中,我们可以看到dubbo-admin如何处理各种异常情况,以及如何记录和上报日志。这部分内容对于理解和优化系统的稳定性和可维护性具有重要意义。 通过深入学习和理解dubbo-admin的源码...

    dubbo入门学习框架源码

    2. 监控报警:结合Dubbo Monitor进行服务性能监控,设置合理的阈值,及时发现并处理异常情况。 3. 微服务治理:结合Spring Cloud或Service Mesh等微服务框架,提升服务治理的灵活性和扩展性。 通过深入学习和实践...

    Spring Boot集成dubbo+zookeeper

    在实际操作中,我们还需要关注一些关键点:如服务的暴露与引用、异常处理、调用超时、负载均衡策略等。此外,Zookeeper的使用需要注意其数据持久化和集群配置,确保服务的高可用性。 最后,测试是验证集成是否成功...

    dubbo-dubbo-2.7.2源码

    Dubbo内置了监控中心,可以收集服务的调用统计、异常信息等,通过可视化界面展示,帮助开发者实时了解服务运行状态,及时发现问题并进行优化。 通过对Dubbo 2.7.2源码的深度解析,我们可以了解到其设计思想和实现...

    dubbo2.0源码解读

    5. **源文件概述**:Dubbo的源代码包括多个模块,如dubbo-common、dubbo-remoting、dubbo-rpc、dubbo-cluster等,分别处理基础工具、网络通信、远程调用、集群逻辑等。 6. **核心机制分析**: - **服务注册与发现*...

    基于zookeeper 监控dubbo provider 下线,提供dubbo 服务下线 邮箱预警

    本项目结合两者,实现了一个针对`Dubbo`服务提供者(`Provider`)下线的监控系统,并提供了邮件预警功能,确保系统能够及时发现并处理服务异常情况,减少因服务不可用导致的业务中断。 1. **Zookeeper在Dubbo中的作用...

    dubbo-dubbo-2.5.10管理平台代码

    包中,可以看到各种过滤器的实现,它们在调用链路中起到了处理异常、控制流量等作用。 6. **配置管理**:Dubbo的配置可以通过`com.alibaba.dubbo.config`包下的类进行管理,如`ApplicationConfig`、`ProviderConfig...

    dubbo本地测试服务

    5. **异常处理**: 检查异常情况,如超时、网络中断等,确保系统有良好的容错机制。 6. **性能测试**: 可选地,进行压力测试,评估服务的性能和稳定性。 七、最佳实践 1. **单元测试**: 对每个接口编写单元测试,...

    dubbo监控中心管理

    2. **服务调用统计**:实时监控服务的调用次数、成功率、平均响应时间等关键指标,帮助识别性能瓶颈和异常情况。 3. **服务配置管理**:允许开发者动态调整服务的配置,如超时时间、重试次数等,无需重启服务即可...

    EShop-SOA:EShop基于Dubbo实现SOA服务化分解,并基于RocketMQ解决了分布式事务(新版SpringBootSOASkeleton)

    所有实体类都必须实现serializable接口Dubbo异常处理机制:异常类和接口类在同一jar包里,直接引发,否则被调方服务中抛出的异常,在调用方中会被包一层RuntimeException,无法获得原来的异常 TCC解决订单支付问题 ...

    dubbo教程代码案例

    2. 异常处理:服务提供者应妥善处理异常,避免直接抛出,而是返回友好的错误信息。 3. 服务版本管理:合理规划服务版本,避免因版本升级导致的兼容性问题。 4. 性能调优:通过调整Dubbo的参数,如超时时间、重试次数...

    dubbo-admin.war.zip

    7. **异常跟踪**:当服务出现异常时,`dubbo-admin`能记录详细的异常信息,帮助开发者快速定位问题。 为了更好地利用`dubbo-admin`,我们需要确保其与Dubbo服务的正确集成。这通常涉及到配置注册中心地址、服务提供...

    Dubbo调用java接口程序

    在实际开发中,还需要考虑服务的版本管理、接口参数校验、异常处理等高级特性,以及如何在微服务环境中有效地使用Dubbo来构建高可用、高并发的分布式系统。对于初学者,理解并熟练掌握这些步骤和概念是至关重要的。...

Global site tag (gtag.js) - Google Analytics