C# Web Service异常处理的两个常见问题

开发 后端
创建用户异常和Web Service的异常处理的问题是C# Web Service异常处理的两个常见问题,本文将向您介绍如何通过SoapExceptionHelper实现一致的异常处理。

在.Net中实现Web服务时,在Web服务接口中产生的任何用户异常(非SoapException之外的异常)都被包装为SoapException传递给客户端,这使得难以采用通常的方式处理Web Service的异常。本文讲述如何通过SoapExceptionHelper实现一致的异常处理。

C# Web Service异常处理问题

在.Net中实现Web服务时,Web服务接口中产生的任何用户异常(非SoapException之外的异常)都被包装为SoapException传递给客户端 ,用户错误信息放置在SoapException的Message属性中。

下面的例子演示了一个SoapException封装的用户异常信息。WebMethod接口TestException代码抛出一个InvalidOperationException:

  1. [WebMethod]   
  2. public void TestException() {   
  3. throw new InvalidOperationException  
  4. ("Invalid Operation.");   
  5. }   

WebMethod的客户端将捕获一个SoapException异常,Message消息如下:

其中Message消息包含一段“...-->[ 1 ]:[ 2 ] at ....”的信息,[1]为用户异常类,[2]为用户异常消息。而一个原始的SoapException(用new SoapException(...)的方式创建并抛出的异常)则没有这些信息,下面是一个原始的SoapException消息:

遗憾的是,目前的SoapException并没有提供更多直接的手段直接获取原来的异常信息,唯一包含的用户异常信息在Message字符串中,对于使用Web Service作为分布式机制的应用系统来说是非常不方便的,调用者无法捕获原来的异常,难以获取用户友好的异常信息。同时,因为Web Service接口代理不再抛出原来的异常,应用的开发者需要考虑两套完全不同的异常处理机制,带来了程序结构的复杂性。

创建SoapException辅助类:SoapExceptionHelper 来解决C# Web Service异常处理

SoapExceptionHelper辅助类包含下列主要接口:

  1. IsUserException:是否是一个UserException   
  2. UserException:返回原始的UserException   
  3. Message:原始异常的错误消息。   
  4. 获得原始的用户异常类和异常消息   
  5. 通过正则表达式类我们可以获得原始的用户异常类和异常消息:   
  6.  
  7. ///    
  8. /// 读取UserException信息。   
  9. /// 
  10.    
  11. private void ReadUserExceptionInfo() {   
  12. //match user exception class   
  13. System.Text.RegularExpressions.  
  14. MatchCollection mc = Regex.Matches  
  15. (soapException.Message, "---> ([^:]+):");   
  16. if (mc.Count >= 1) {   
  17. userExceptionClass = mc[0].Groups[1].Value;   
  18. //match user exception message   
  19. mc = Regex.Matches(soapException.  
  20. Message, "---> [^:]+:(.*)\n");   
  21. if (mc.Count > 0) UserExceptionMessage   
  22. = mc[0].Groups[1].Value;   
  23. }   
  24. }   
  25.  
  26. 创建用户异常实例   
  27. UserException接口利用反射机制创建  
  28. 一个原来的Exception类实例:   
  29.  
  30. ... ...   
  31. Assembly callingAssemply = Assembly.  
  32. GetCallingAssembly();   
  33. Type exceptionType = GetExceptionType  
  34. (callingAssemply); //获得用户异常类型定义   
  35. Exception e = null;   
  36. try {   
  37. try {   
  38. e = Activator.CreateInstance(exceptionType,   
  39. new object[]{UserExceptionMessage},   
  40. nullas Exception;   
  41. }   
  42. catch {}   
  43. //if no exists constructor with message parameter,   
  44. use no parameters constructor.   
  45. if (e == null) e = Activator.CreateInstance  
  46. (exceptionType) as Exception;   
  47. }catch(Exception ex) {   
  48. throw new SoapExceptionHelperException  
  49. (userExceptionClass, ex);   
  50. }   
  51.  
  52. return e;   

创建用户异常的问题

因为用户异常可能定义在不同的集成块中,SoapExceptionHelper可能无法知道它的位置,无法正确的获取C# Web Service异常处理类型,如一个与SoapExceptionHelper所在集成块和调用集成块(CallingAssembly)不再同一个引用范围内的异常类。SoapExceptionHelper如果无法创建原始异常的实例,就创建一个System.Exception对象实例。

为了创建真正的原始异常类,调用者可以在外部获得实际的异常类型,并传递给SoapExceptionHelper,因为调用者可以明确的引用异常定义 所在的集成块。示例如下:

  1. // 项目引用中引入异常定义所在的集成块   
  2. ...   
  3. SoapExceptionHelper helper =   
  4. new SoapExceptionHelper(se);   
  5. Type type = Type.GetType(helper.  
  6. UserExceptionClass, "<异常类所在的集成块>");   
  7. Exception e = helper.GetUserException(type);   
  8.  
  9. 如果外部没有传递异常类型定义,  
  10. SoapExceptionHelper尝试以以下顺序获取异常类型定义:   
  11. Executing Assembly   
  12. Calling Assembly   
  13. Referenced Assemblies (of Calling Assembly)   
  14. System.Exception   
  15. 使用SoapExceptionHelper   
  16. 返回用户友好的消息   
  17. 使用SoapExceptionHelper显示示例1中的错误消息:   
  18. try {   
  19. ... ... // call web method   
  20. } catch (SoapException se){   
  21. MessageBox.Show(new SoapExceptionHelper(se).  
  22. Message) ; //show "Invalid Operation." string   

【编辑推荐】

  1. C#中定义装箱和拆箱详解
  2. 浅谈C#类型系统
  3. 三种不同的C#异常类型
  4. 详细介绍C#编译器
  5. C#异常机制的相关解释
责任编辑:冰荷 来源: jpinw
相关推荐

2015-05-15 09:37:24

iOS开发争议

2022-07-01 13:17:13

Docker

2016-12-22 15:31:47

数据中心监控自动化警报

2009-08-20 15:38:50

C#建立Web Ser

2009-08-11 13:27:22

C#创建Web Ser

2013-05-23 10:23:04

DebianDebian 7.0

2020-05-25 22:41:27

LoRaWAN物联网技术物联网

2009-06-15 17:05:03

C#基元类型

2009-08-05 18:28:05

C#异常处理

2009-11-05 12:45:25

WCF异常

2009-08-31 17:26:32

C#异常处理

2009-08-06 14:08:08

C#语言异常处理

2013-01-06 10:43:07

Linux集群

2009-08-12 16:37:22

C#变量类型转换

2009-08-11 11:07:49

Java调用C# we

2019-09-25 14:48:03

区块链以太坊比特币

2014-08-25 10:24:01

Linux

2009-08-11 13:07:26

C#类库中添加Web

2015-11-16 09:12:40

android问题开发

2011-02-22 14:00:16

vsftpd
点赞
收藏

51CTO技术栈公众号