详解ASP.NET MVC数据验证的一个特殊方法
创始人
2024-06-06 19:20:17
0

这里我们将介绍ASP.NET MVC数据验证实现的一个特殊方法,包括数据的验证,验证后数据的提交等等。51CTO编辑推荐《ASP.NET MVC框架视频教程》。

关于ASP.NET MVC数据验证,用起来很特别,因为MS的封装,使人理解起来很费解。也可能很多人都在Scott Guthrie等人写的一本《ASP.NET MVC 1.0》书中,见过NerdDinner项目中对Dinner对象修改和添加的时的数据验证。但有许多封装的地方,不知道是怎样的工作原理,今天研究了,拿出来给大家分享一下。

数据库还是上一篇blog中的库与表,同样的方法来创建news表的实体类,在自动生成的news这个实体类中,我们发现有一个特殊的分部方法:

  1. partial void OnValidate(System.Data.Linq.ChangeAction action); 

这个方法没有实现,我们根据C#的语法知道,如果分部类中的分部方法,没有实现的话,调用和定议的地方都不会起什么作用。现在,我们要去完善这个方法,让它“用”起来。

首先,人产在Models中创建news类的另一部分,代码如下:

  1.     public partial  class news  
  2.     {  
  3.         partial void OnValidate(System.Data.Linq.ChangeAction action)  
  4.         {  
  5.             if (!IsValid)  
  6.             {  
  7.                 throw new ApplicationException("验证内容项出错!");  
  8.             }  
  9.         }  
  10.         public bool IsValid  
  11.         {  
  12.             get { return (GetRuleViolations().Count() == 0); }  
  13.         }  
  14.         public IEnumerable GetRuleViolations()  
  15.         {  
  16.             if (String.IsNullOrEmpty(this.title .Trim () ))  
  17.                 yield return new RuleViolation("题目步能为空!", "题目");  
  18.             if (String.IsNullOrEmpty(this.contents .Trim ()))  
  19.                 yield return new RuleViolation("内容不能为空!", "内容");            
  20.             yield break;  
  21.         }  
  22.     }  
  23. ///  
  24.     /// 规则信息类  
  25.     /// 
  26.  
  27.     public class RuleViolation  
  28.     {  
  29.         public string ErrorMessage { get; private set; }  
  30.         public string PropertyName { get; private set; }  
  31.    
  32.         public RuleViolation(string errorMessage)  
  33.         {  
  34.             ErrorMessage = errorMessage;  
  35.         }  
  36.    
  37.         public RuleViolation(string errorMessage, string propertyName)  
  38.         {  
  39.             ErrorMessage = errorMessage;  
  40.             PropertyName = propertyName;  
  41.         }  
  42.     } 

在这里给出这么多代码,其实是提前有设计的,因为从业务角度考虑,还不应该写这部分代码。RuleViolation类很简单,就是一个包括了两个属性的类(这个类的结构设计是根据后面的ModelState.AddModelError主法来设计的)。

在news分部类中,有一个IsValid的属性,这个属性是bool类型的,返回值取决于GetRuleViolations这个方法,这个方法返回值是一个IEnumerable类型的,IEnumerable是通过news的几个属性是否为空来生成跌代的。如果title或contents为Null或””,就返回跌代。其实真正的用户数据的验证就是在这里实现,用户的数据的对与错,就是一个逻辑,只要用户数据不符合规则,就可以 “yield return new RuleViolation("错误标识","错误提示信息!")”;这里的错误码提示信息是显示到客户端的,所以要处理好友好的提示。

现在验证用户数据,生成错误列表的工作都做完了,但关键是怎么能让用户提交数据时,调用OnValidate。这个问题,先放一下,请记住,上面的代码,只要在用户提交数据时,调用OnValidate,这样就能得到错误集合。

现在,让我们来处理Cotroller和View层,在Cotroller层,首先来添加index这个Action,代码如下:

  1. public ActionResult Index()  
  2.         {             
  3.             var NewsList = DCDC.news.Select(newss=>newss);  
  4.             return View(NewsList );  
  5.      } 

这个Action返回所有news表中的记录。

对应的View如下:

  1. <%@ Page Title="" Language="C#" MasterPageFile="~/Views/Shared/Site.Master" Inherits="System.Web.Mvc.ViewPage>" %> 
  2.  ID="Content1" ContentPlaceHolderID="TitleContent" runat="server"> 
  3.      Index  
  4.  
  5.  ID="Content2" ContentPlaceHolderID="MainContent" runat="server"> 
  6.    
  7.     

    Index

     
  8.    
  9.      
  10.         
  11.  
  12.             
  13.  
  14.             
  15.  
  16.             
  17.  
  18.             
  19.  
  20.             
  21.  
  22.             
  23.  
  24.         
  25.  
  26.    
  27.     <% foreach (var item in Model) { %> 
  28.       
  29.         
  30.  
  31.             
  32.  
  33.             
  34.  
  35.             
  36.  
  37.             
  38.  
  39.             
  40.  
  41.             
  42.  
  43.         
  44.     
  45.     <% } %> 
  46.     
  47.  
  48.                 ID  
  49.             
  50.  
  51.                 title  
  52.             
  53.  
  54.                 datetimes  
  55.             
  56.  
  57.                 contents  
  58.             
  59.  
  60.                 IsValid  
  61.             
  62.  
  63.                 <%= Html.ActionLink("Edit", "Edit", new { id=item.ID }) %> |  
  64.                 <%= Html.ActionLink("Details", "Details", new { id=item.ID })%> 
  65.             
  66.  
  67.                 <%= Html.Encode(item.ID) %> 
  68.             
  69.  
  70.                 <%= Html.Encode(item.title) %> 
  71.             
  72.  
  73.                 <%= Html.Encode(String.Format("{0:g}", item.datetimes)) %> 
  74.             
  75.  
  76.                 <%= Html.Encode(item.contents) %> 
  77.             
  78.  
  79.                 <%= Html.Encode(item.IsValid) %> 
  80.             
  81.  
  82.     

     

  83.         <%= Html.ActionLink("Create New", "Create") %> 
  84.     

     
  85.  

代码中,需要我们注意是的    <%= Html.ActionLink("Edit", "Edit", new { id=item.ID }) %>

因为要导航到Edit的View,把以接下来我们创建Edit的Action和View(因为在编辑数据时,要用到验证,Edit才是我们的重点)。

  1. public ActionResult Edit(int id)  
  2.         {  
  3.             var list = DCDC.news.Single(newss=>newss.ID ==id);  
  4.             return View(list);  
  5.      } 

<%= Html.ActionLink("Edit", "Edit", new { id=item.ID }) %>中的id会被当成参数送到EditController的Edit(int id)的Action,成为Edit方法的实参。
Edit.aspx页面如下图:

Edit.aspx页面

 

对应Edit的Action生成view,代码如下:

  1. <%@ Page Title="" Language="C#" MasterPageFile="~/Views/Shared/Site.Master" Inherits="System.Web.Mvc.ViewPage" %> 
  2.  ID="Content1" ContentPlaceHolderID="TitleContent" runat="server"> 
  3.      编辑  
  4.  
  5.  ID="Content2" ContentPlaceHolderID="MainContent" runat="server"> 
  6.      style ="text-align :left ;">编辑 
  7.     <%= Html.ValidationSummary("Edit was unsuccessful. Please correct the errors and try again.") %> 
  8.     <% using (Html.BeginForm())  
  9.        { %> 
  10.         
     
  11.             详细内容       
  12.             

     

  13.                  for="title">标题: 
  14.                 <%= Html.TextBox("title", Model.title) %> 
  15.                 <%= Html.ValidationMessage("题目", "*")%> 
  16.             

     
  17.             

     

  18.                  for="datetimes">时间: 
  19.                 <%= Html.TextBox("datetimes", String.Format("{0:g}", Model.datetimes)) %> 
  20.                  <%= Html.ValidationMessage("时间", "*") %> 
  21.             

     
  22.             

     

  23.                  for="contents">内容: 
  24.                 <%= Html.TextBox("contents", Model.contents) %> 
  25.                 <%= Html.ValidationMessage("内容", "*")%> 
  26.             

     
  27.             

     

  28.                  type="submit" value="更新" /> 
  29.             

     
  30.         
  31.  
  32.     <% } %> 
  33.     
     
  34.         <%=Html.ActionLink("Back to List", "Index") %> 
  35.     
 
  •  
  • 如果要单击“更新”返回数据新数据,还需要我们写如下一个Action:

    1. [AcceptVerbs(HttpVerbs.Post)]  
    2.         public ActionResult Edit(int id,FormCollection formValuews)  
    3.         {  
    4.             news Sig_news = DCDC.news.Single(newss => newss.ID == id);  
    5.             try 
    6.             {                 
    7.                 Sig_news.title = formValuews.GetValue("title").AttemptedValue;  
    8.                 Sig_news.datetimes = DateTime.Parse(formValuews.GetValue("datetimes").AttemptedValue);  
    9.                 Sig_news.contents = formValuews.GetValue("contents").AttemptedValue;  
    10.                 DCDC.SubmitChanges();  
    11.                 return RedirectToAction("Index");  
    12.             }  
    13.             catch 
    14.             {  
    15.                 foreach (var v in Sig_news.GetRuleViolations())  
    16.                 {  
    17.                     ModelState.AddModelError(v.PropertyName,v.ErrorMessage);  
    18.                 }  
    19.                 return View(Sig_news);  
    20.             }  
    21.         } 

    这个Edit的Action是用户提交返来更新数据库的,我们可以从formValuews得到用户在页面上更新的数据,来更新Sig_news对象,然后调用DCDC.SubmitChanges();去更新数据库,如果没有民常,会导航到index.aspx页面。如果发生异常,就会运行到catch里。如果还记得,在本文的前半部分,我们说到OnValidate,是数据在提交时应该验证,但在这里,我们并没有显示的调用OnValidate这个方法,但实际运行中,我们发现,这个方法被执行了,如果我们建立跟踪,把断点设在DCDC.SubmitChanges();如果我们数据有民常,会发现当DCDC.SubmitChanges();执行完后就会跳到partial void OnValidate(System.Data.Linq.ChangeAction action)这个方法,这是怎么做到的呢?我们猜测,一定是在数据提交时,调用OnValidate这个方法。为了找到它们的关系,只好用Reflector.exe来“探测”一下了(Reflector.exe的用法就不说了)。

    SubmitChanges方法是DataContext的一个方法,这个类位于System.Data.Linq命空间下,用Reflector.exe打开SubmitChanges,看到this.SubmitChanges(ConflictMode.FailOnFirstConflict);定位这个方法,可以看到new ChangeProcessor(this.services, this).SubmitChanges(failureMode);定位查找会发现ValidateAll(orderedList);在这个方法中,多处看到  SendOnValidate(obj2.Type, obj2, ChangeAction.Insert);这个方法,再定位,有这样一行代码  type.OnValidateMethod.Invoke(item.Current, new object[] { changeAction });这里,正是通过反射调用了OnValidate这个方法。这样我们就找到了SubmitChanges执行时调用OnValidate的方法了(其不用调用OnValidate也可以验证用户数据,只需要写个方法,在SubmitChanges 提交以前执行就可以达到同样效果)。同时,当发生异常时,OnValidate会抛出一个Application的异常,这里会被public ActionResult Edit(int id,FormCollection formValuews)方法中的Catch捕获到,就执行如下代码:

    1. foreach (var v in Sig_news.GetRuleViolations())  
    2.                 {  
    3.                     this.ModelState.AddModelError(v.PropertyName,v.ErrorMessage);  
    4.                 }  
    5.                 return View(Sig_news); 

    这行代码的意思是把错误的信息,以键值的方式放入ModelState中,ModelState是一个ModelStateDictionary类型,这个类型实现了IDictionary, ICollection>, IEnumerable>, IEnumerable这些接口(这里要注意,ModelState是当前对象的一个属性,并且它的AddModelError方法的***个参数key有其独特的作用)。处理完异常后,还是返回当前页面。这时你会发现,在页面的   <%= Html.ValidationSummary("Edit was unsuccessful. Please correct the errors and try again.") %>发生了变化,把我们错误的地方去提示出来了,这里就是,为什么我们把错误信息放到ModelState中,而错误则显示在了Html.ValidationSummary中了呢?并且发生错误的数据后会加上了一个红色的“*”,这是怎么样做到的呢?

    再次利用Reflector.exe,查看Html.ValidationSummary方法和Html.ValidationMessage方法,会发现它们显示的数据是从ModelState 中获取的,如果ModelState 这个集合中没有数据,Html.ValidationSummary和Html.ValidationMessage就返回空,如果发生异常,this.ModelState中有子项,就会通过Html.ValidationSummary和Html.ValidationMessage在页面页上显示出来。因为Html.ValidationMessage在页面上有多个,所以在this.ModelState.AddModelError(v.PropertyName,v.ErrorMessage);方法中的v.PropertyName就有了用处了,这个值要与<%= Html.ValidationMessage("题目", "*")%>中的***个参数对应,这样<%= Html.ValidationMessage("题目", "*")%>才能起到作用,显示出第二个参数“*”。
    这样一来,就达到了ASP.NET MVC的数据验证。由于ASP.NET MVC验证拐的弯比较多,所以下来用个图来说明一下。

    ASP.NET MVC验证图

    详解ASP.NET MVC数据验证的一个特殊方法就介绍到这里。

    【编辑推荐】

    1. 详解ASP.NET MVC分页的实现方法
    2. ASP.NET MVC与WebForm区别谈
    3. ASP.NET MVC应用程序执行过程分析
    4. ASP.NET MVC分页控件的实现
    5. 有关ASP.NET MVC框架的一些基础知识

    相关内容

    热门资讯

    如何允许远程连接到MySQL数... [[277004]]【51CTO.com快译】默认情况下,MySQL服务器仅侦听来自localhos...
    如何利用交换机和端口设置来管理... 在网络管理中,总是有些人让管理员头疼。下面我们就将介绍一下一个网管员利用交换机以及端口设置等来进行D...
    施耐德电气数据中心整体解决方案... 近日,全球能效管理专家施耐德电气正式启动大型体验活动“能效中国行——2012卡车巡展”,作为该活动的...
    20个非常棒的扁平设计免费资源 Apple设备的平面图标PSD免费平板UI 平板UI套件24平图标Freen平板UI套件PSD径向平...
    德国电信门户网站可实时显示全球... 德国电信周三推出一个门户网站,直观地实时提供其安装在全球各地的传感器网络检测到的网络攻击状况。该网站...
    为啥国人偏爱 Mybatis,... 关于 SQL 和 ORM 的争论,永远都不会终止,我也一直在思考这个问题。昨天又跟群里的小伙伴进行...
    《非诚勿扰》红人闫凤娇被曝厕所... 【51CTO.com 综合消息360安全专家提醒说,“闫凤娇”、“非诚勿扰”已经被黑客盯上成为了“木...
    2012年第四季度互联网状况报... [[71653]]  北京时间4月25日消息,据国外媒体报道,全球知名的云平台公司Akamai Te...