成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

ASP.NET MVC基于異常處理的解決方案

開發 后端
本文我將通過它的擴展實現與EntLib的集成,并提供一個完整的解決異常處理解決方案。

  EntLib的異常處理應用塊(Exception Handling Application Block)是一個不錯的異常處理框架,它使我們可以采用配置的方式來定義異常處理策略。而ASP.NET MVC是一個***可擴展開發框架,在這篇文章中我將通過它的擴展實現與EntLib的集成,并提供一個完整的解決異常處理解決方案。  

目錄

  一、基本異常處理策略

  二、通過自定義Action處理異常

  三、通過配置的Error View處理異常

  四、自定義ActionInvoker:ExceptionActionInvoker

  五、自定義Controller:BaseController

  一、基本異常處理策略

  我們首先來討論我們的解決方案具體采用的異常處理策略:

  對于執行Controller的某個Action方法拋出的異常,我們會按照指定配置策略進行處理。我們可以采取日志記錄、異常替換和封裝這些常用的異常處理方式;

  對于處理后的異常,如果異常處理策略規定需要將其拋出,則會自動重定向到與異常類型匹配的出錯頁面。我們會維護一個異常類型和Error View的匹配關系;

  對于處理后的異常,如果異常處理策略規定不需要將其拋出,則會執行與當前Action操作相匹配的錯誤處理Action進行處理。異常處理Action方法默認采用“On{Action}Error”這樣的命名規則,而當前上下文會與異常處理操作方法的參數進行綁定。除次之外,我們會設置當前ModelState的錯誤信息;

  如果用戶不曾定義相應的異常處理Action,依然采用“錯誤頁面重定向”方式進行異常處理。

  二、通過自定義Action處理異常

  為了讓讀者對上面介紹的異常處理頁面有一個深刻的理解,我們來進行一個實例演示。該實例用于模擬用戶登錄,我們定義了如下一個只包含用戶名和密碼兩個屬性的Model:LoginInfoModel。

  1. namespaceArtech.Mvc.ExceptionHandling.Models  
  2. {  
  3. publicclassLoginInfo  
  4. {  
  5. [Display(Name ="User Name")]  
  6. [Required(ErrorMessage = "User Name is manadatory!")]  
  7. publicstringUserName { getset; }  
  8. [Display(Name = "Password")]  
  9. [DataType(DataType.Password)]  
  10. [Required(ErrorMessage = "Password is manadatory!")]  
  11. publicstringPassword { getset; }  
  12. }  

  我們定義了如下一個AccountController,它是我們自定義的BaseController的子類。AccountController在構造的時候調用基類構造函數指定的參數代表異常處理策略的配置名稱。SignIn方法代表用于進行“登錄”的操作,而OnSignInError就表示該操作對應的異常處理操作。如果在SignIn操作中拋出的異常經過處理后無需再拋出,則會通過調用OnSignInError,而此時ModelState已經被設置了相應的錯誤消息。

  1. publicclassAccountController : BaseController  
  2. {  
  3. publicAccountController()  
  4.  base("myPolicy")5:{ }  
  5. publicActionResult SignIn()  
  6. {  
  7. returnView(newLoginInfo());  
  8. }  
  9. [HttpPost]  
  10. publicActionResult SignIn(LoginInfo loginInfo)  
  11. {  
  12. if(!ModelState.IsValid)  
  13. {  
  14. returnthis.View(newLoginInfo { UserName = loginInfo.UserName });  
  15. }  
  16. if(loginInfo.UserName != "Foo")  
  17. {  
  18. thrownewInvalidUserNameException();  
  19. }  
  20. if(loginInfo.Password != "password")  
  21. {  
  22. thrownewUserNamePasswordNotMatchException();  
  23. }  
  24. ViewBag.Message = "Authentication Succeeds!";  
  25. returnthis.View(newLoginInfo { UserName = loginInfo.UserName });  
  26. }  
  27. publicActionResult OnSignInError(stringuserName)  
  28. {  
  29. returnthis.View(newLoginInfo { UserName = userName });  
  30. }  

  具體定義在SignIn操作方法中的認證邏輯是這樣的:如果用戶名不是“Foo”則拋出InvalidUserNameException異常;如果密碼不是“password”則拋出UserNamePasswordNotMatchException異常。下面是SignIn操作對應的View的定義:

  1. @model Artech.Mvc.ExceptionHandling.Models.LoginInfo  
  2. @{  
  3. ViewBag.Title = "SignIn";  
  4. }  
  5. @Html.ValidationSummary()  
  6. @if (ViewBag.Messages != null)  
  7. {   
  8. @ViewBag.Messages  
  9. }  
  10. @using (Html.BeginForm())  
  11. {   
  12. @Html.EditorForModel()<inputtype="submit"value="SignIn"/>  

  在AccountController初始化時指定的異常處理策略“myPolicy”定義在如下的配置中。我們專門針對SignIn操作方法拋出的InvalidUserNameException和UserNamePasswordNotMatchException進行了處理,而ErrorMessageSettingHandler是我們自定義的異常處理器,它僅僅用于設置錯誤消息。如下面的代碼片斷所示,如果上述的這兩種類型的異常被拋出,最終的錯誤消息會被指定為“User name does not exist!”和“User name does not match password!”。

  1. <exceptionHandling> 
  2. <exceptionPolicies> 
  3. <addnameaddname="myPolicy"> 
  4. <exceptionTypes> 
  5. <addnameaddname="InvalidUserNameException" 
  6. type="Artech.Mvc.ExceptionHandling.Models.InvalidUserNameException, Artech.Mvc.ExceptionHandling" 
  7. postHandlingAction="None"> 
  8. <exceptionHandlers> 
  9. <addnameaddname="ErrorMessageSettingHandler" 
  10. type="Artech.Mvc.ExceptionHandling.ErrorMessageSettingHandler, Artech.Mvc.ExceptionHandling" 
  11. errorMessage="User name does not exist!"/> 
  12. </exceptionHandlers> 
  13. </add> 
  14. <addnameaddname="UserNamePasswordNotMatchException" 
  15. type="Artech.Mvc.ExceptionHandling.Models.UserNamePasswordNotMatchException, Artech.Mvc.ExceptionHandling" 
  16. postHandlingAction="None"> 
  17. <exceptionHandlers> 
  18. <addnameaddname="ErrorMessageSettingHandler" 
  19. type="Artech.Mvc.ExceptionHandling.ErrorMessageSettingHandler, Artech.Mvc.ExceptionHandling" 
  20. errorMessage="User name does not match password!"/> 
  21. </exceptionHandlers> 
  22. </add> 
  23. </exceptionTypes> 
  24. </add> 
  25. </exceptionPolicies> 
  26. </exceptionHandling> 

  現在我們通過路由映射將AccountController和Sign設置為默認Controller和Action后,開啟我們的應用程序。在輸入錯誤的用戶名和錯誤明碼的情況下在ValidationSummary中將自動得到相應的錯誤消息。

三、通過配置的Error View處理異常

  在上面的配置中,針對InvalidUserNameException和UserNamePasswordNotMatchException這兩種異常類型的配置策略都將PostHandlingAction屬性設置為“None”,意味著不會將原來的異常和處理后的異常進行重新拋出。現在我們將該屬性設置為“ThrowNewException”,意味著我們會將處理后的異常重新拋出來。

  1. <exceptionHandling> 
  2. <exceptionPolicies> 
  3. <addnameaddname="myPolicy"> 
  4. <exceptionTypes> 
  5. <addnameaddname="InvalidUserNameException"type="Artech.Mvc.ExceptionHandling.Models.InvalidUserNameException, Artech.Mvc.ExceptionHandling"  
  6. postHandlingAction="ThrowNewException"> 
  7. ...  
  8. <addnameaddname="UserNamePasswordNotMatchException"type="Artech.Mvc.ExceptionHandling.Models.UserNamePasswordNotMatchException, Artech.Mvc.ExceptionHandling"9:postHandlingAction="ThrowNewException"> 
  9. ...  
  10. </add> 
  11. </exceptionTypes> 
  12. </add> 
  13. </exceptionPolicies> 
  14. </exceptionHandling> 

  按照我們上面的異常處理策略,在這種情況下我們將采用“錯誤頁面”的方式來進行異常處理。也HandleErrorAttribute的處理方式類似,我們支持異常類型和Error View之間的匹配關系,而這是通過類似于如下的配置來定義的。值得一提的是,這里的異常類型是經過處理后重新拋出的異常

  1. <exceptionHandling> 
  2. <exceptionPolicies> 
  3. <addnameaddname="myPolicy"> 
  4. <exceptionTypes> 
  5. <addnameaddname="InvalidUserNameException"type="Artech.Mvc.ExceptionHandling.Models.InvalidUserNameException, Artech.Mvc.ExceptionHandling"  
  6. postHandlingAction="ThrowNewException"> 
  7. ...  
  8. <addnameaddname="UserNamePasswordNotMatchException"type="Artech.Mvc.ExceptionHandling.Models.UserNamePasswordNotMatchException, Artech.Mvc.ExceptionHandling"9:postHandlingAction="ThrowNewException"> 
  9. ...  
  10. </add> 
  11. </exceptionTypes> 
  12. </add> 
  13. </exceptionPolicies> 
  14. </exceptionHandling> 

  如上面的配置所示,我們為InvalidUserNameException和UserNamePasswordNotMatchException這兩種異常類型定義了不同的Error View,分別是“InvalideUserNameError”和“UserNamePasswordNotMatchError”,詳細定義如下所示:

  1. @{  
  2. Layout = null;  
  3. }  
  4. <!DOCTYPEhtml> 
  5. <html> 
  6. <head> 
  7. <title>Error</title> 
  8. </head> 
  9. <body> 
  10. <pstylepstyle="color:Red; font-weight:bold">Sorry,the user name you specify does not exist!</p> 
  11. </body> 
  12. </html> 
  13.  @{  
  14. Layout = null;  
  15. }  
  16. <!DOCTYPEhtml> 
  17. <html> 
  18. <head> 
  19. <title>Error</title> 
  20. </head> 
  21. <body> 
  22. <pstylepstyle="color:Red; font-weight:bold">Sorry, The password does not match the given user name!</p> 
  23. </body> 
  24. </html> 

  現在我們按照上面的方式運行我們的程序,在分別輸入錯誤的用戶名和密碼的情況下會自動顯現相應的錯誤頁面。

四、自定義ActionInvoker:ExceptionActionInvoker

  對于上述的兩種不同的異常處理方式最終是通過自定義的ActionInvoker來實現的,我們將其命名為ExceptionActionInvoker。如下面的代碼片斷所式,ExceptionActionInvoker直接繼承自ControllerActionInvoker。屬性ExceptionPolicy是一個基于指定的異常策略名稱創建的ExceptionPolicyImpl 對象,用于針對EntLib進行的異常處理。而屬性GetErrorView是一個用于獲得作為錯誤頁面的ViewResult對象的委托。整個異常處理的核心定義在InvokeAction方法中,該方法中指定的handleErrorActionName參數代表的是“異常處理操作名稱”,整個方法就是按照上述的異常處理策略實現的。

 

  1. usingSystem;  
  2. usingSystem.Collections.Generic;  
  3. usingSystem.Linq;  
  4. usingSystem.Web;  
  5. usingSystem.Web.Mvc;  
  6. usingArtech.Mvc.ExceptionHandling.Configuration;  
  7. usingMicrosoft.Practices.EnterpriseLibrary.Common.Configuration;  
  8. usingMicrosoft.Practices.EnterpriseLibrary.ExceptionHandling;  
  9. namespaceArtech.Mvc.ExceptionHandling  
  10. {  
  11. publicclassExceptionActionInvoker: ControllerActionInvoker  
  12. {  
  13. protectedExceptionHandlingSettings ExceptionHandlingSettings{get; privateset;}  
  14. protectedvirtualFunc<string, HandleErrorInfo, ViewResult> GetErrorView { get; privateset; }  
  15. publicExceptionPolicyImpl ExceptionPolicy { get; privateset; }  
  16. publicExceptionActionInvoker(stringexceptionPolicy,Func<string, HandleErrorInfo, ViewResult> getErrorView)  
  17. {  
  18. this.ExceptionPolicy = EnterpriseLibraryContainer.Current.GetInstance<ExceptionPolicyImpl>(exceptionPolicy);  
  19. this.GetErrorView = getErrorView;  
  20. this.ExceptionHandlingSettings = ExceptionHandlingSettings.GetSection();  
  21. }  
  22. publicoverrideboolInvokeAction(ControllerContext controllerContext, stringhandleErrorActionName)  
  23. {  
  24. ExceptionContext exceptionContext = controllerContext asExceptionContext;  
  25. if(null== exceptionContext)  
  26. {  
  27. thrownewArgumentException("The controllerContext must be ExceptionContext!""controllerContext");  
  28. }  
  29. try 
  30. {  
  31. exceptionContext.ExceptionHandled = true;  
  32. if(this.ExceptionPolicy.HandleException(exceptionContext.Exception))  
  33. {  
  34. HandleRethrownException(exceptionContext);  
  35. }  
  36. else 
  37. {  
  38. if(ExceptionHandlingContext.Current.Errors.Count == 0)  
  39. {  
  40. ExceptionHandlingContext.Current.Errors.Add(exceptionContext.Exception.Message);  
  41. }  
  42. ControllerDescriptor controllerDescriptor = this.GetControllerDescriptor(exceptionContext);  
  43. ActionDescriptor handleErrorAction = FindAction(exceptionContext, controllerDescriptor, handleErrorActionName);  
  44. if(null!= handleErrorAction)  
  45. {  
  46. IDictionary<stringobject>parameters = GetParameterValues(controllerContext, handleErrorAction);  
  47. exceptionContext.Result = this.InvokeActionMethod(exceptionContext, handleErrorAction, parameters);  
  48. }  
  49. else 
  50. {  
  51. HandleRethrownException(exceptionContext);  
  52. }  
  53. }  
  54. returntrue;  
  55. }  
  56. catch(Exception ex)  
  57. {  
  58. exceptionContext.Exception = ex;60:HandleRethrownException(exceptionContext);  
  59. returntrue;  
  60. }  
  61. }  
  62. protectedvirtualvoidHandleRethrownException(ExceptionContext exceptionContext)  
  63. {  
  64. stringerrorViewName = this.GetErrorViewName(exceptionContext.Exception.GetType());  
  65. stringcontrollerName = (string)exceptionContext.RouteData.GetRequiredString("controller");  
  66. stringaction = (string)exceptionContext.RouteData.GetRequiredString("action");  
  67. HandleErrorInfo handleErrorInfo = newHandleErrorInfo(exceptionContext.Exception, controllerName, action);70:exceptionContext.Result = this.GetErrorView(errorViewName, handleErrorInfo);  
  68. }  
  69. protectedstringGetErrorViewName(Type exceptionType)  
  70. {  
  71. ExceptionErrorViewElement element = ExceptionHandlingSettings.ExceptionErrorViews75:.Cast<ExceptionErrorViewElement>().FirstOrDefault(el=>el.ExceptionType == exceptionType);  
  72. if(null!= element)  
  73. {  
  74. returnelement.ErrorView;  
  75. }  
  76. if(null== element &&null!= exceptionType.BaseType!= null)  
  77. {  
  78. returnGetErrorViewName(exceptionType.BaseType);  
  79. }  
  80. else 
  81. {  
  82. return"Error";  
  83. }  
  84. }  
  85. }  

  五、自定義Controller:BaseController

  ExceptionActionInvoker最終在我們自定義的Controller基類BaseController中被調用的。ExceptionActionInvoker對象在構造函數中被初始化,并在重寫的OnException方法中被調用。

 

  1. usingSystem;  
  2. usingSystem.Web.Mvc;  
  3. namespaceArtech.Mvc.ExceptionHandling  
  4. {  
  5. publicabstractclassBaseController : Controller  
  6. {  
  7. publicBaseController(stringexceptionPolicy)  
  8. {  
  9. Func<string, HandleErrorInfo, ViewResult> getErrorView = (viewName, handleErrorInfo) => this.View(viewName, handleErrorInfo);  
  10. this.ExceptionActionInvoker = newExceptionActionInvoker(exceptionPolicy,getErrorView);  
  11. }  
  12. publicBaseController(ExceptionActionInvoker actionInvoker)  
  13. {  
  14. this.ExceptionActionInvoker = actionInvoker;
  15. }
  16. publicvirtualExceptionActionInvoker ExceptionActionInvoker { get; privateset; }  
  17. protectedvirtualstringGetHandleErrorActionName(stringactionName)  
  18. {  
  19. returnstring.Format("On{0}Error", actionName);  
  20. }  
  21. protectedoverridevoidOnException(ExceptionContext filterContext)  
  22. {  
  23. using(ExceptionHandlingContextScope contextScope = newExceptionHandlingContextScope(filterContext))  
  24. {  
  25. stringactionName = RouteData.GetRequiredString("action");  
  26. stringhandleErrorActionName = this.GetHandleErrorActionName(actionName);  
  27. this.ExceptionActionInvoker.InvokeAction(filterContext, handleErrorActionName);  
  28. foreach(var error inExceptionHandlingContext.Current.Errors)  
  29. {  
  30. ModelState.AddModelError(Guid.NewGuid().ToString() ,error.ErrorMessage);  
  31. }  
  32. }  
  33. }  
  34. }  

  值得一提的是:整個OnException方法中的操作都在一個ExceptionHandlingContextScope中進行的。顧名思義, 我們通過ExceptionHandlingContextScope為ExceptionHandlingContext創建了一個范圍。ExceptionHandlingContext定義如下,我們可以通過它獲得當前的ExceptionContext和ModelErrorCollection,而靜態屬性Current返回當前的ExceptionHandlingContext對象。

 

  1. publicclassExceptionHandlingContext  
  2. {  
  3. [ThreadStatic]  
  4. privatestaticExceptionHandlingContext current;  
  5. publicExceptionContext ExceptionContext { get; privateset; }  
  6. publicModelErrorCollection Errors { get; privateset; }  
  7. publicExceptionHandlingContext(ExceptionContext exceptionContext)  
  8. {  
  9. this.ExceptionContext = exceptionContext;
  10. this.Errors = newModelErrorCollection();  
  11. }  
  12. publicstaticExceptionHandlingContext Current  
  13. {  
  14. get { returncurrent; }  
  15. set { current = value;}  
  16. }  

  在BaseController的OnException方法中,當執行了ExceptionActionInvoker的InvokeAction之后,我們會將當前ExceptionHandlingContext的ModelError轉移到當前的ModelState中。這就是為什么我們會通過ValidationSummary顯示錯誤信息的原因。對于我們的例子來說,錯誤消息的指定是通過如下所示的ErrorMessageSettingHandler 實現的,而它僅僅將指定的錯誤消息添加到當前ExceptionHandlingContext的Errors屬性集合中而已。

 

  1. [ConfigurationElementType(typeof(ErrorMessageSettingHandlerData))]  
  2. publicclassErrorMessageSettingHandler : IExceptionHandler  
  3. {  
  4. publicstringErrorMessage { get; privateset; }  
  5. publicErrorMessageSettingHandler(stringerrorMessage)  
  6. {  
  7. this.ErrorMessage = errorMessage;  
  8. }  
  9. publicException HandleException(Exception exception, Guid handlingInstanceId)  
  10. {  
  11. if(null== ExceptionHandlingContext.Current)  
  12. {  
  13. thrownewInvalidOperationException("...");  
  14. }  
  15. if(string.IsNullOrEmpty(this.ErrorMessage))  
  16. {  
  17. ExceptionHandlingContext.Current.Errors.Add(exception.Message);  
  18. }  
  19. else 
  20. {  
  21. ExceptionHandlingContext.Current.Errors.Add(this.ErrorMessage);  
  22. }  
  23. returnexception;  
  24. }  

原文鏈接:http://www.cnblogs.com/artech/archive/2012/01/10/exception-handling-in-asp-mvc.html

【編輯推薦】

  1. ASP.NET MVC會成為ASP.NET未來主流的UI平臺?
  2. ASP.NET性能優化之構建自定義文件緩存
  3. 想愛容易相處難:當ASP.NET MVC愛上IoC
  4. 導致ASP.NET站點重啟的十大原因
  5. 淺談ASP.NET核心對象
責任編輯:彭凡 來源: 博客園
相關推薦

2009-07-22 11:00:30

ASP.NET MVC

2009-07-24 11:24:33

ASP.NET中文亂碼

2009-02-06 14:11:36

ASP.NET.NET全局異常處理

2009-07-22 17:37:06

ASP.NET Ses

2009-07-31 12:43:59

ASP.NET MVC

2009-07-24 13:20:44

MVC框架ASP.NET

2010-06-04 19:06:47

連接MySQL數據庫

2010-09-02 15:18:42

CSSASP.NET

2009-07-23 16:53:17

ASP.NET中文變問

2009-07-30 16:02:53

2009-07-20 15:44:32

ASP.NET MVC

2009-07-22 09:11:02

Action方法ASP.NET MVC

2009-07-22 13:24:24

ASP.NET MVC

2009-07-22 10:09:59

ASP.NET MVC

2009-07-23 14:31:20

ASP.NET MVC

2009-07-23 15:44:39

ASP.NET MVC

2009-07-20 10:53:59

ASP.NET MVC

2009-07-22 10:34:37

ActionInvokASP.NET MVC

2011-09-22 10:58:56

ASP.NET

2009-07-20 12:59:53

ASP.NET MVCASP.NET框架的功
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 天天夜碰日日摸日日澡 | 99国内精品久久久久久久 | 黄色片av| 国产成人在线观看免费 | 中文字幕免费视频 | 精品免费国产一区二区三区四区介绍 | 久久国产精品精品国产色婷婷 | 日韩亚洲一区二区 | 国产精品一区二区三区久久久 | 五月激情综合 | 日本天堂一区二区 | 999国产精品视频免费 | 337p日本欧洲亚洲大胆精蜜臀 | 91偷拍精品一区二区三区 | 自拍偷拍第一页 | 99精品网 | 日韩一区精品 | 黄色一级免费 | 久久性色 | 日韩精品无码一区二区三区 | 国产毛片久久久久久久久春天 | 欧美精品久久久久久 | 日韩国产欧美视频 | 日韩精品视频在线观看一区二区三区 | 成人欧美一区二区三区黑人孕妇 | 亚洲欧美日韩成人在线 | 日韩中文在线观看 | 久久综合爱| a级片网站 | 午夜一区二区三区在线观看 | 羞羞的视频免费观看 | 精品国产乱码久久久久久蜜臀 | 日韩毛片视频 | 日韩精品免费在线观看 | 欧美视频成人 | 男女免费在线观看视频 | 国产精品1区2区 | 午夜一区 | 91色在线| 艹逼网 | 日韩1区|