您的位置:首页 > 编程语言 > ASP

asp.net MVC学习系列-前后台数据校验

2018-02-08 18:29 232 查看
转载 http://blog.csdn.net/ydm19891101/article/details/43485253
Web页面,是一个数据进入的接口,对数据的合法性、正确性检测非常重要。做好表单数据验证工作可以防止那些攻击数据以及不合理数据。通过使用客户端和服务器端双重验证,可以保证表单数据的安全性。有时候客户端验证不是太安全的,可以设法绕开客户端验证,所以服务端验证也是需要的。但是为了减轻服务器的压力,还是尽量使用客户端验证。下面分别介绍MVC设计模式中如何实现两种验证。

一、客户端校验

客户端的校验有两种方式:jQuery方式、MVC方式
在讲校验之前先说明一个问题:VIew视图的创建必须是强类型的。
首先在Models文件夹下添加一个类,以Teacher类为例,具体代码:

[csharp] view plain copypublic class Teacher  
{  
    public string Tno { set; get; }  
    public string Tname { set; get; }  
    public int Tsex { set; get; }  
    public string Taddress { set; get; }  
    public string Tphone { set; get; }  
}  
重新生成项目
新添加一个Teacher控制器,为该控制器添加一个Create的Action,添加对应于该Action的视图(选择Teacher强类型视图),默认为我们生成了以下代码

[csharp] view plain copy<% using (Html.BeginForm("Create", "Teacher", FormMethod.Post, new { id = "frmAdd" }))//这段代码需要我们填写Url  
   { %>  
    <%: Html.ValidationSummary(true)%>  
    <fieldset>  
        <legend>Teacher</legend>  
  
        <div class="editor-label">//由于我们选择的是强类型视图,后台返回的数据都是Teacher实体类型即model实体  
            <%: Html.LabelFor(model => model.Tno)%>//Tno与我们Teacher类的Tno属性是对应的  
        </div>  
        <div class="editor-field">  
            <%: Html.EditorFor(model => model.Tno)%>  
            <%: Html.ValidationMessageFor(model => model.Tno)%>  
        </div>  
  
        <div class="editor-label">  
            <%: Html.LabelFor(model => model.Tname)%>  
        </div>  
        <div class="editor-field">  
            <%: Html.EditorFor(model => model.Tname)%>  
            <%: Html.ValidationMessageFor(model => model.Tname)%>  
        </div>  
  
        <div class="editor-label">  
            <%: Html.LabelFor(model => model.Tsex)%>  
        </div>  
        <div class="editor-field">  
            <%: Html.EditorFor(model => model.Tsex)%>  
            <%: Html.ValidationMessageFor(model => model.Tsex)%>  
        </div>  
  
        <div class="editor-label">  
            <%: Html.LabelFor(model => model.Taddress)%>  
        </div>  
        <div class="editor-field">  
            <%: Html.EditorFor(model => model.Taddress)%>  
            <%: Html.ValidationMessageFor(model => model.Taddress)%>  
        </div>  
  
        <div class="editor-label">  
            <%: Html.LabelFor(model => model.Tphone)%>  
        </div>  
        <div class="editor-field">  
            <%: Html.EditorFor(model => model.Tphone)%>  
            <%: Html.ValidationMessageFor(model => model.Tphone)%>  
        </div>  
  
        <p>  
            <input type="submit" value="Create" />  
        </p>  
    </fieldset>  
<% } %>  
  
<div>  
    <%: Html.ActionLink("Back to List", "Index") %>  
</div>  
如果我们这个时候就运行程序,查看页面源代码时可以看到微软已经为我们自动添加了一些校验:

[html] view plain copy<div class="editor-field">  
                <input class="text-box single-line" id="Tphone" name="Tphone" type="text" value="">  
                <span class="field-validation-valid" data-valmsg-for="Tphone" data-valmsg-replace="true"></span>  
            </div>  
我们在定义Teacher类时定义了Tphone属性为字符串类型,这样在我们输入时就通过class="field-validation-valid"限定了我们只能输入字符串。这种验证简单但是有一定局限性。下面就讲解我们自定义的两种验证方式。

1.jQuery验证方式添加script脚本引用:
[html] view plain copy<script type="text/javascript" src="../../Scripts/jquery-1.4.4.js"></script>  
<script src="../../Scripts/jquery.validate.js" type="text/javascript"></script>     
添加jQuery进行数据验证

[javascript] view plain copy<script type="text/javascript">  
       $(function () {  
           $("#frmAdd").validate({  
               rules: {  
                   Tsex: {  
                   required:true,  
                   digits: true  
                   }  
               },  
               messages: {  
                   Tsex:"*必须输入0或者1"  
               }  
           });  
       });  
   </script>  
2.MVC验证方式方式1:Model层校验
在Model层上实现这些验证规则的好处是,这将确保这些验证在应用中任何使用Teacher对象的场景中都被执行,这将帮助确保我们的代码DRY(Don't Repeat Yourself,不重复自己,意味着我们应该只在一处实施验证规则,然后使我们的控制器,Action方法和视图来遵守),避免在多处重复这些规则。MVC3允许开发人员轻松地在模型或视图模型类上添加声明式验证特性,然后在应用中实施模型绑定操作时,这些验证规则就会被自动执行。
添加script脚本引用:

[html] view plain copy<script type="text/javascript" src="../../Scripts/jquery-1.4.4.js"></script>  
<script src="../../Scripts/jquery.validate.js" type="text/javascript"></script>   
<script src="../../Scripts/jquery.validate.unobtrusive.js" type="text/javascript"></script>  

修改Teacher类代码

[csharp] view plain copy[StringLength(10, MinimumLength = 6)]  
[Required(ErrorMessage = "工号为必填项")]  
public string Tno { set; get; }  
[StringLength(10, MinimumLength = 2)]  
[Required(ErrorMessage = "姓名为必填项")]  
public string Tname { set; get; }  
[Range(0, 1)]  
[Required(ErrorMessage = "性别为必填项")]  
public int Tsex { set; get; }  
[StringLength(30, MinimumLength = 6)]  
public string Taddress { set; get; }  
[RegularExpression(@"^\d+$", ErrorMessage = "*电话必须是数字")]  
public string Tphone { set; get; }  
运行程序,在加载页面之前,微软会先在程序集的元数据上添加以上的验证标签,加载页面时也会读取元数据,当发现元数据中存在验证标签时会为该标签添加上验证标签属性,具体代码如下所示:

[html] view plain copy<fieldset>  
            <legend>Teacher</legend>  
      
            <div class="editor-label">  
                <label for="Tno">Tno</label>  
            </div>  
            <div class="editor-field">  
                <input class="text-box single-line" data-val="true" data-val-length="字段 Tno 必须是一个字符串,其最小长度为 6,最大长度为 10。" data-val-length-max="10" data-val-length-min="6" data-val-required="工号为必填项" id="Tno" name="Tno" type="text" value="">  
                <span class="field-validation-valid" data-valmsg-for="Tno" data-valmsg-replace="true"></span>  
            </div>  
      
            <div class="editor-label">  
                <label for="Tname">Tname</label>  
            </div>  
            <div class="editor-field">  
                <input class="text-box single-line" data-val="true" data-val-length="字段 Tname 必须是一个字符串,其最小长度为 2,最大长度为 10。" data-val-required="姓名为必填项" id="Tname" name="Tname" type="text" value="">  
                <span class="field-validation-valid" data-valmsg-for="Tname" data-valmsg-replace="true"></span>  
            </div>  
      
            <div class="editor-label">  
                <label for="Tsex">Tsex</label>  
            </div>  
            <div class="editor-field">  
                <input class="text-box single-line" data-val="true" data-val-number="The field Tsex must be a number." data-val-range="字段 Tsex 必须在 0 和 1 之间。" data-val-range-max="1" data-val-range-min="0" data-val-required="性别为必填项" id="Tsex" name="Tsex" type="text" value="">  
                <span class="field-validation-valid" data-valmsg-for="Tsex" data-valmsg-replace="true"></span>  
            </div>  
      
            <div class="editor-label">  
                <label for="Taddress">Taddress</label>  
            </div>  
            <div class="editor-field">  
                <input class="text-box single-line" data-val="true" data-val-length="字段 Taddress 必须是一个字符串,其最小长度为 6,最大长度为 30。" data-val-length-max="30" data-val-length-min="6" id="Taddress" name="Taddress" type="text" value="">  
                <span class="field-validation-valid" data-valmsg-for="Taddress" data-valmsg-replace="true"></span>  
            </div>  
      
            <div class="editor-label">  
                <label for="Tphone">Tphone</label>  
            </div>  
            <div class="editor-field">  
                <input class="text-box single-line" data-val="true" data-val-regex="*电话必须是数字" data-val-regex-pattern="^\d+$" id="Tphone" name="Tphone" type="text" value="">  
                <span class="field-validation-valid" data-valmsg-for="Tphone" data-valmsg-replace="true"></span>  
            </div>  
      
            <p>  
                <input type="submit" value="Create">  
            </p>  
        </fieldset>  
Model层的校验方式比较简单,容易理解。Model层的数据校验主要有四种:必填项、正则匹配、限定范围、字符串长度,具体使用如上所示。

方式2:EF框架实体类校验
第二种实现方式与第一种差不多,也是在模型对应的实体类上添加上面的四种校验方式。但是,当我们根据模型更新实体的属性时,会造成之前添加的验证的丢失,这是我们不愿看到的。解决方法就是利用微软为我们提供的一种伙伴类技术,具体操作如下。
Teacher类代码还是用之前的

[csharp] view plain copypublic class Teacher  
{  
    public string Tno { set; get; }  
    public string Tname { set; get; }  
    public int Tsex { set; get; }  
    public string Taddress { set; get; }  
    public string Tphone { set; get; }  
}  
在Models文件夹下添加TeacherExt类,修改其代码如下

[csharp] view plain copy    [MetadataType(typeof(TeacheExt))]//代表:Teacher要共享TeacherExt的元数据信息  
    public partial class Teacher  
    {  
  
  
    }  
    public class TeacheExt  
    {    //元数据描述:   Tname属性有特性:StringLength  
        [Required(ErrorMessage = "姓名为必填项")]  
        [StringLength(20, ErrorMessage = "请限制输入字符串长度小于20")]  
        public string Tname { set; get; }  
    }  

经过上面的操作,Teacher类的Tname具有了数据校验,并且当更新Teacher实体类的属性时,其Tname属性仍具有了数据校验特效。
当我们加载页面时微软也为我们的Tname标签添加了以下的标签属性

[html] view plain copy<input class="text-box single-line" data-val="true" data-val-length="请限制输入字符串长度小于20" data-val-length-max="20" data-val-required="姓名为必填项" id="Tname" name="Tname" type="text" value="">  

上面讲解了客户端的两种验证方式,客户端的校验只能保证用户的正确输入,并不能保证数据就一定能提交到数据库,这时候就需要后台校验了。

二、服务器端校验

MVC设计模式同样为我们提供了后台校验数据的方法,具体使用如下:
我们还是需要先在Teacher类添加一些验证特效,具体代码如下

[csharp] view plain copy[StringLength(10, MinimumLength = 6)]  
      [Required(ErrorMessage = "工号为必填项")]  
      public string Tno { set; get; }  
      [StringLength(10, MinimumLength = 2)]  
      [Required(ErrorMessage = "姓名为必填项")]  
      public string Tname { set; get; }  
      [Range(0, 1)]  
      [Required(ErrorMessage = "性别为必填项")]  
      public int Tsex { set; get; }  
      [StringLength(30, MinimumLength = 6)]  
      public string Taddress { set; get; }  
      [RegularExpression(@"^\d+$", ErrorMessage = "*电话必须是数字")]  
      public string Tphone { set; get; }  
相应的修改Action方法的代码

[csharp] view plain copypublic ActionResult Create(MvcLogin.Models.Teacher teacher)  
      {  
          if (ModelState.IsValid)  //校验成功  
          {  
              //执行数据的入库、更新等操作  
          }  
          else  
          {  
              //给出提示信息  
          }  
          return View(teacher);  
      }  

在我们的HTML表单被提交回服务器时,上面的Create方法就会被调用。因为该Action方法接受一个“Teacher” 对象为参数,ASP.NET MVC会创建一个Teacher对象,自动地将进来的表单输入数值映射到该对象上。作为该过程的一部分,ASP.NET MVC还会检查该Teacher对象上的DataAnnotation验证特性是否合法。如果一切都合法,那么我们代码中的 ModelState.IsValid检查就会返回真值,在这种情形下,我们(最终)将把该Teacher对象保存到数据库中。但如果Teacher对象上有任何验证错误的话,我们的Action方法就会以该不合法Teacher对象的数据重新显示表单,这是通过上面代码片段中最后一行代码实现的。然 后,错误消息就会显示在我们的视图中,因为我们的Create表单在每一个<%= Html.TextBoxFor() %>辅助方法的调用旁边都有一个<%= Html.ValidationMessageFor() %>辅助方法调用。Html.ValidationMessageFor() 辅助方法会针对传入视图的任何不合法的模型属性输出合适的错误消息。

这个模式/方式有一个好处,就是非常容易配置,它还允许我们轻松地添加或改变我们Teacher类上的验证规则,而不必改变控制器或视图中的任何代码。这个在一个地方指定验证规则,然后在所有的地方都会被遵守的能力,允许我们快速地发展我们的应用和规则,并且将代码保持在非常DRY的程度。
MVC学习系列-了解Url的路由规则

MVC学习系列-HTML帮助器的使用

MVC学习系列-表单的异步提交
MVC学习系列-表单数据提交

MVC学习系列-ViewData与ViewBag

版权声明:本文为博主原创文章,未经博主允许不得转载。如果感觉文章哪里写的不对或者存在疑问,欢迎留言,共同学习、进步!
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: