因为要导航到
Edit
的
View
,把以接下来我们创建
Edit
的
Action
和
View(
因为在编辑数据时,要用到验证,
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
会被当成参数送到
EditController
的
Edit(int id)
的
Action
,成为
Edit
方法的实参。
Edit.aspx
页面如下图:
对应
Edit
的
Action
生成
view
,代码如下:
<%
@
Page Title="" Language="C#" MasterPageFile="~/Views/Shared/Site.Master" Inherits="System.Web.Mvc.ViewPage<MvcCompany.Models.news>" %>
<
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);
}
}
这个
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
捕获到,就执行如下代码:
foreach
(var v in Sig_news.GetRuleViolations())
{
this.ModelState.AddModelError(v.PropertyName,v.ErrorMessage);
}
return View(Sig_news);
这行代码的意思是把错误的信息,以键值的方式放入
ModelState
中,
ModelState
是一个
ModelStateDictionary
类型,这个类型实现了
IDictionary<string, ModelState>, ICollection<KeyValuePair<string, ModelState>>, IEnumerable<KeyValuePair<string, ModelState>>, 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
的数据验证。由于
ASPNET MVC
验证捌的弯比较多,所以下来用个图来说明一下。