自定义异常的一个简单应用

在写N层结构应用程序时候经常得涉及到层次间方法的调用.调用具有返回值的方法时,象方法A调用方法B,正常情况下方法B应该返回正常值,但若是方法B找不到满足方法A的结果集或是有异常情况发生,这时我们一般采用特殊值来返回,以来标示方法B返回的不是正常值,然后在方法A中根据B中返回的特殊值做相应的处理.

   对于这种情况,我们要在方法A和方法B中同时做处理,返回的这个特殊值是方法A和方法B的共同约定值,两个方法约定好在异常或没预料情况发生时方法B应该返回什么值,同时方法A也应该知道这个值是什么,因为它需要知道当方法B返回给它这个值时预示着什么问题发生,然后再做相应处理.这样处理的话你会发觉这两方法联系太紧密了,如果很多方法间都这样处理,这样的话层次间藕合度也加大了, 这并不是我们期望看到的.

   我们可以考虑用自定义异常来解决这样问题,参考了 宝玉Asp.Net Forums中的自定义异常类 .自定义一个异常类CustomException,在处理上面提到情况时,只要在方法B中抛出CustomException(type)异常,在方法A中catch这个异常,然后通过调用这个被catch异常的message属性就可以获知我们自定义的异常信息并把它展示给最终用户.而且通过使用这种方法,你也会发觉有些原本需要返回值的方法现在不需要了,象是有些方法是用来向数据库中添加数据,一般我们会在添加成功后返回true,添加失败返回false.但用自定义异常后,只需要在添加失败后抛出一个CustomException(type),然后在适当地方catch就可以了,免去返回值和一些判断返回值的逻辑.

   很多人可能会对这样做带来的性能问题有所顾虑,Applied Microsoft.Net Framework Programming中这样提到异常处理和常用的异常报告(HRESULT,特殊返回值代码,等等)之间的性能差别.里面说"比较异常处理和常用的异常报告(HRESULT,特殊返回值代码,等等)之间的性能差别通常是很困难的.如果在每次方法调用时我们都要自己编写代码来检查它们的返回值,并将其返回给调用者,那么我们的应用程序性能将会受到严重的影响.就算不考虑性能,我们必须为此编写的额外代码和潜在的错误也会大幅度增加.异常处理相对来说则是一种更好的选择".
   下面还只是简单实现错误信息提示,还没有异常修复或回滚.只是起个抛砖引玉.更多的以后还会继续下去.

CustomException.cs
/// <summary>
    
/// 自定义异常类CustomException
    
/// </summary>

     public   class  CustomException:ApplicationException
    
{
        
//记录异常的类型
        private CustomExceptionType exceptionType;

        
public CustomException(CustomExceptionType type):base()
        
{
            
this.exceptionType=type;
        }


        
public CustomException(CustomExceptionType type,string message):base(message)
        
{
            
this.exceptionType=type;
        }


        
//序列化
        public override void GetObjectData(SerializationInfo info,StreamingContext context)
        
{
            
base.GetObjectData(info,context);
        }


        
//重写message方法,以让它显示相应异常提示信息
        public override string Message
        
{
            
get
            
{
                
//根据异常类型从message.xml中读取相应异常提示信息
         
        return string.Format(XmlMessageManager.GetXmlMessage((int)exceptionType),base.Message);

                }

        }

    }

BlogExceptionType枚举
     public   enum  BlogExceptionType
    
{
        UserNameError
=3,
        UserPasswordError
=4,
        ValidateCodeDisabled
=5,
        UserLoginDisabled 
= 6,
        UserPasswordChangeFailed 
= 7,
        UnknownError 
= 8
    }


XmlMessageMananger.cs
     /// <summary>
    
/// XmlMessageManager.cs
    
/// </summary>

     public   class  XmlMessageManager
    
{
        
// 根据id获取相对应自定义异常提示信息
        public static string GetXmlMessage(int id)
        
{
            XmlDocument xmlDoc
=new XmlDocument();
            xmlDoc.Load(HttpContext.Current.Server.MapPath(
"..//ColBlog//ColBlog.Component//messages.xml"));

            XmlNodeList nodeList
=xmlDoc.SelectSingleNode("root").ChildNodes;

            
foreach(XmlNode node in nodeList)
            
{
                XmlElement xe
=(XmlElement)node;
                
                
if(xe.GetAttribute("id").ToString()==id.ToString())
                
{
                    
return xe.SelectSingleNode("body").InnerText;
                }

            }

            
return string.Empty;
        }

    }

message.xml
<!--  message.xml  -->
<? xml version="1.0" encoding="utf-8"  ?>  
< root >

< message  id ="3" >
< body > 用户名只能由英文字母,数字,下划线组成 </ body >
</ message >

< message  id ="4" >
< body > 密码长度必须在5-20之间 </ body >
</ message >

< message  id ="5" >
< body > 您输入的验证码不匹配,请重新返回输入 </ body >
</ message >

< message  id ="6" >
< body > 用户登陆失败,请重新输入 </ body >
</ message >

< message  id ="7" >
< body > 更改密码时出现错误,请您再试一次 </ body >
</ message >

< message  id ="8" >
< body > 很抱歉,发生了未知错误 </ body >
</ message >

</ root >


如果自定义异常类中定义了新的字段,我们必须让其实现ISerializable接口的GetObjectData(SerializationInfo info,StreamingContext context)方法和一个特殊的受保护的构造器CustomException(SerializationInfo info,StreamingContext context).其中GetObjectData()方法用来序列化新定义字段以及基类(ApplicatonException)定义的字段,而特殊构造器是为反序列化新定义的字段和基类定义字段。
象是CustomException类中若定义一个新字段field1.我们应该在CustomException类中实现GetObjectData方法和CustomException(SerializationInfo info,StreamingContext context)构造器.

           private   string field1;         

            // 序列化
         public   override   void  GetObjectData(SerializationInfo info,StreamingContext context)
        
{
            info.AddValue(
"Field1",field1);

            
base.GetObjectData(info,context);
        }


        
// 反序列化
         protected  CustomException(SerializationInfo info,StreamingContext context): base (info,context)
        
{
            field1
=info.GetString("Field1");
        }


.NET自定义异常的一个简单应用

转自:

http://inrie.cnblogs.com/archive/2005/11/07/270723.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值