在写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,特殊返回值代码,等等)之间的性能差别通常是很困难的.如果在每次方法调用时我们都要自己编写代码来检查它们的返回值,并将其返回给调用者,那么我们的应用程序性能将会受到严重的影响.就算不考虑性能,我们必须为此编写的额外代码和潜在的错误也会大幅度增加.异常处理相对来说则是一种更好的选择".
BlogExceptionType枚举
XmlMessageMananger.cs
message.xml
如果自定义异常类中定义了新的字段,我们必须让其实现ISerializable接口的GetObjectData(SerializationInfo info,StreamingContext context)方法和一个特殊的受保护的构造器CustomException(SerializationInfo info,StreamingContext context).其中GetObjectData()方法用来序列化新定义字段以及基类(ApplicatonException)定义的字段,而特殊构造器是为反序列化新定义的字段和基类定义字段。
象是CustomException类中若定义一个新字段field1.我们应该在CustomException类中实现GetObjectData方法和CustomException(SerializationInfo info,StreamingContext context)构造器.
对于这种情况,我们要在方法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
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);
}
}
}
/// 自定义异常类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
}
{
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;
}
}
/// 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 >
<? 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");
}
// 序列化
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");
}
转自: