因为要导航到EditView,把以接下来我们创建EditActionView(因为在编辑数据时,要用到验证,Edit才是我们的重点)
       public ActionResult Edit(int id)
        {
            var list = DCDC.news.Single(newss=>newss.ID ==id);
            return View(list);
     }
<%= Html.ActionLink("Edit", "Edit", new { id=item.ID }) %>中的id会被当成参数送到EditControllerEdit(int id)Action,成为Edit方法的实参。
Edit.aspx页面如下图:
ASP.NET MVC数据验证(下)_第1张图片
对应EditAction生成view,代码如下:
<% @ Page Title="" Language="C#" MasterPageFile="~/Views/Shared/Site.Master" Inherits="System.Web.Mvc.ViewPage" %>
< asp : Content ID="Content1" ContentPlaceHolderID="TitleContent" runat="server">
     编辑
asp : Content >
< asp : Content ID="Content2" ContentPlaceHolderID="MainContent" runat="server">
    <h2 style ="text-align :left ;"> 编辑h2>
    <%= Html.ValidationSummary("Edit was unsuccessful. Please correct the errors and try again.") %>
    <% using (Html.BeginForm())
       { %>
        <fieldset>
            <legend> 详细内容legend>    
            <p>
                <label for="title"> 标题:label>
                <%= Html.TextBox("title", Model.title) %>
                <%= Html.ValidationMessage(" 题目" , "*")%>
            p>
            <p>
                <label for="datetimes"> 时间:label>
                <%= Html.TextBox("datetimes", String.Format("{0:g}", Model.datetimes)) %>
                 <%= Html.ValidationMessage(" 时间" , "*") %>
            p>
            <p>
                <label for="contents"> 内容:label>
                <%= Html.TextBox("contents", Model.contents) %>
                <%= Html.ValidationMessage(" 内容" , "*")%>
            p>
            <p>
                <input type="submit" value=" 更新" />
            p>
        fieldset>
    <% } %>
    <div>
        <%=Html.ActionLink("Back to List", "Index") %>
    div>
 
asp : Content >
如果要单击“更新”返回数据新数据,还需要我们写如下一个Action
[AcceptVerbs(HttpVerbs.Post)]
        public ActionResult Edit(int id,FormCollection formValuews)
        {
            news Sig_news = DCDC.news.Single(newss => newss.ID == id);
            try
            {              
                Sig_news.title = formValuews.GetValue("title").AttemptedValue;
                Sig_news.datetimes = DateTime.Parse(formValuews.GetValue("datetimes").AttemptedValue);
                Sig_news.contents = formValuews.GetValue("contents").AttemptedValue;
                DCDC.SubmitChanges();
                return RedirectToAction("Index");
            }
            catch
            {
                foreach (var v in Sig_news.GetRuleViolations())
                {
                    ModelState.AddModelError(v.PropertyName,v.ErrorMessage);
                }
                return View(Sig_news);
            }
        }
这个EditAction是用户提交返来更新数据库的,我们可以从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捕获到,就执行如下代码:
foreach (var v in Sig_news.GetRuleViolations())
                {
                    this.ModelState.AddModelError(v.PropertyName,v.ErrorMessage);
                }
                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.ValidationSummaryHtml.ValidationMessage就返回空,如果发生异常,this.ModelState中有子项,就会通过Html.ValidationSummaryHtml.ValidationMessage在页面页上显示出来。因为Html.ValidationMessage在页面上有多个,所以在this.ModelState.AddModelError(v.PropertyName,v.ErrorMessage);方法中的v.PropertyName就有了用处了,这个值要与<%= Html.ValidationMessage("题目", "*")%>中的第一个参数对应,这样<%= Html.ValidationMessage("题目", "*")%>才能起到作用,显示出第二个参数“*”。
这样一来,就达到了 ASP.NET MVC 的数据验证。由于 ASPNET MVC  验证捌的弯比较多,所以下来用个图来说明一下。
ASP.NET MVC数据验证(下)_第2张图片