ASP.NET MVC 入門8、ModelState與數據驗證


原帖地址:http://www.cnblogs.com/QLeelulu/archive/2008/10/08/1305962.html

ViewData有一個ModelState的屬性,這是一個類型為ModelStateDictionary的ModelState類型的字典集合。在進行數據驗證的時候這個屬性是比較實用的。在使用Html.ValidationMessage()的時候,就是從ViewData.ModelState中檢測是否有指定的KEY。假設存在。就提示錯誤信息。比如在前一篇文章ASP.NET MVC 入門7、Hellper與數據的提交與綁定中使用到的UpdateModel方法:

image

我們在View中使用Html.ValidationMessage(string modelName)來對指定的屬性進行驗證:

image

Html.ValidationMessage()有幾個重載:

image

當中ValidationSummary()是用於顯示所有的驗證信息的。跟ASP.NET里面的ValidationSummary驗證控件差點兒相同。

我們測試一下/Admin/Setting頁面:

image

在用UpdateModel方法更新BlogSettings.Instance.PostsPerPage的時候,當我們如圖所看到的填寫"10d"的時候。因為PostsPerPage為整型的,所以UpdateModel方法就會出錯,同一時候會往ViewData.ModelState加入對應的錯誤信息。從而Html.ValidationMessage()方法就能夠從ViewData.ModelState中檢測到錯誤並提示。

同一時候Html.ValidationMessage()方法會為出錯的屬性的輸入框加入一個名為"input-validation-error"的CSS類,同一時候后面的提示信息的CSS類名為"field-validation-error":

image

CSS類的樣式是能夠由我們自己自由定義的。如上圖的紅色高亮顯示。

好,以下我們來實現發表新隨筆的功能。我們先寫一個提供用戶輸入隨筆內容的表單頁面:

復制代碼
< p >
   
< label for ="Title" > 標題 </ label >
   
<% = Html.TextBox( " Title " , new { id =   " Title " , @class =   " required " }) %>
   
<% = Html.ValidationMessage( " Title " ) %>
</ p >
< p >
   
< label for ="Content" > 內容 </ label >
   
<% = Html.TextArea( " Content " ) %>
   
<% = Html.ValidationMessage( " Content " ) %>
</ p >
< p >
   
< label for ="Slug" > URL地址別名(假設為空則和標題同名) </ label >
   
<% = Html.TextBox( " Slug " , new { id =   " Slug " , @class =   " required " }) %>
   
<% = Html.ValidationMessage( " Slug " ) %>
</ p >  
復制代碼

 

然后我們對用戶提交過來的數據進行保存:

復制代碼
[AcceptVerbs( " POST " ), ActionName( " NewPost " )]
public ActionResult SaveNewPost(FormCollection form)
{
    Post post
=   new Post();

   
try
    {
        UpdateModel(post,
new [] { " Title " , " Content " , " Slug " });
    }
   
catch
    {
       
return View(post);
    }

    post.Save();
   
return ShowMsg( new List < string > () { " 發表新隨筆成功 " });
}
復制代碼

 

因為這三個值都是字符串類型,所以假設值為空的話,UpdateModel也是不會出錯的,而我們的Title和Content是不同意為空的。或者我們想我們的Slug的長度不能超過100,也就是須要有我們自己的業務規則。

這時候我們也許會這樣寫:

復制代碼
try
{
    UpdateModel(post,
new [] { " Title " , " Content " , " Slug " });
}
catch
{
   
return View(post);
}

if ( string .IsNullOrEmpty(post.Title))
{
    ViewData.ModelState.AddModelError(
" Title " , post.Title, " 標題不能為空 " );
}
if ( string .IsNullOrEmpty(post.Content))
{
    ViewData.ModelState.AddModelError(
" Content " , post.Content, " 內容不能為空 " );
}

if ( ! ViewData.ModelState.IsValid)
{
   
return View(post);
}
復制代碼

 

ViewData.ModelState提供了一個AddModelError的方法。方便我們加入驗證失敗的信息。我們能夠如上代碼這樣進行對象的業務規則驗證,可是一旦業務規則多了。這種代碼是非常壯觀的。並且不好控制。

那么我們該怎么更好的進行業務規則的驗證呢?得意於BlogEngine.Net的良好架構,我們能夠非常輕松的完畢這一點。

首先。讓我們改動一下BlogEngine.Core里面BusinessBase的代碼。我們前面說過。BusinessBase實現了IDataErrorInfo接口,該接口有個索引器,導致ViewData.Eval()方法調用時搜索索引器的值時返回String.Empty而使ViewData.Eval()覺得是找到值了,從而失效。

image

我們能夠將return string.Empty改動為return null。

但我們這里並不須要用到這個接口。所以我們把該接口去掉,並把對應的代碼凝視了。然后我們再暴露一個BrokenRules的屬性,用於返回當前的全部破壞性業務規則(紅框部分代碼為我們加入的):

image

BusinessBase提供了一個抽象的ValidationRules方法,用於在業務類重寫這種方法往里面加入驗證規則(詳細請看BusinessBase的Validation節)。

Validation

 

我們在Post類中重寫這種方法來加入驗證規則:

image

然后我們能夠在Controller的Action中非常優雅的書寫我們的代碼來進行業務規則的驗證:

復制代碼
[AcceptVerbs( " POST " ), ActionName( " NewPost " )]
public ActionResult SaveNewPost(FormCollection form)
{
    Post post
=   new Post();

   
try
    {
        UpdateModel(post,
new [] { " Title " , " Content " , " Slug " });
    }
   
catch
    {
       
return View(post);
    }

   
if ( ! post.IsValid)
    {
       
foreach ( string key in post.BrokenRules.Keys)
        {
            ViewData.ModelState.AddModelError(key, form[key], post.BrokenRules[key]);
        }
       
return View(post);
    }

    post.Save();
   
return ShowMsg( new List < string > () { " 發表新隨筆成功 " });
}
復制代碼

 

我們注意到上面的Action中用到了一個FormCollection 的參數,這個參數系統會自己主動將Form提交過來的所有表單值(Request.Form)賦給它的。client驗證能夠用jQuery的驗證插件來。這里就不羅嗦了。



免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM