博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
MVC 模型绑定
阅读量:6089 次
发布时间:2019-06-20

本文共 4975 字,大约阅读时间需要 16 分钟。

  在WebForm,获取提交表单的值一般都是Request.Form["Title"]这样的方式。在MVC中,提供了模型绑定机制。让后台获取表单或Url中的参数变得更加简单。

一、基本模型绑定

  你可以直接在参数中用字符串,整型变量,实体或者是List<实体>的方式获取表单提交的参数。

  参数中的这些东西都是与表单中的Html控件的name属性一一对应的。

public ActionResult PersonAdd(int Id)        {            return View();        }

  例如以上代码,它能够匹配Url中的Id参数。如以下两种方法Id都能够匹配到1

  http://localhost/Home/PersonAdd/1  http://localhost/Home/PersonAdd?Id=1

  在例如如下代码:

public ActionResult PersonAdd(string Name)        {            return View();        }

  它能够匹配到表单中提交的张三:

  也能够匹配到Get请求的路径参数:

http://localhost/Home/PersonAdd?Name=张三

  如果是用实体,则会检查该实体的属性名与表单中name属性中对应的标签的值。

  例如有如下实体:

public class Person_Model    {        public int Id { get; set; }        public string Name { get; set; }    }

  在Controller中的参数填写如下:

[HttpPost]        public ActionResult PersonAdd(Person_Model model)        {            if (ModelState.IsValid)  //此处仅作演示,不考虑安全性            {                //插入数据库省略                return Redirect("/Home/PersonManager");            }            return View();        }

  这样的话,模型绑定器会自动检查该实体的属性与Name一一对应的标签并绑定。如下表单的值将被绑定到model实体的属性中。

    

二、显式模型绑定

  UpdateModel与TryUpdateModel都用于显示模型绑定。如果绑定期间出现错误或者模型是无效的。

  UpdateModel将抛出一个异常。因此UpdateModel要用try catch语句块包起来,而TryUpdateModel不会抛出异常,而是返回一个布尔类型的值,true表示绑定成功,false表示绑定失败。如:

[HttpPost]        public ActionResult PersonAdd()        {            Person_Model model = new Person_Model();            try            {                UpdateModel(model);                //插入数据库                return Redirect("/Home/PersonManager");            }            catch            {                return View(model);            }        }

  TruUpdateModel:

[HttpPost]        public ActionResult PersonAdd()        {            Person_Model model = new Person_Model();            if (TryUpdateModel(model))            {                //插入数据库                return Redirect("/Home/PersonManager");            }            else            {                return View(model);            }        }

  另外,模型绑定还有一个模型状态,模型绑定器一斤模型中的每一个值在模型状态中都有相应的一条记录。可以随时查看绑定状态。如:

[HttpPost]        public ActionResult PersonAdd()        {            Person_Model model = new Person_Model();            TryUpdateModel(model);            if (ModelState.IsValid)            {                //if(ModelState.IsValidField("Name"))                //插入数据库                return Redirect("/Home/PersonManager");            }            else            {                return View(model);            }        }

三、安全问题:重复提交

   假设有如下实体:

public class Comment    {        public int Id { get; set; }        //评论者姓名        public string Name { get; set; }        //评论内容        public string Content { get; set; }        //是否已审核        public bool Approved { get; set; }    }

  在Controller中:

public ActionResult CommentAdd(Comment com)        {            if (ModelState.IsValid)            {                //添加数据库                return Redirect("/Home/CommentManager");            }            else            {                 return View(com);            }        }

  在以上代码中,如果有恶意用户在表单数据中添加"Approved=true"来干预表单的提交,那么该评论将是默认就通过审核的。这时候我们可以使用Bind特性来防御重复提交攻击。

  白名单:

[Bind(Include="Name,Content")]      //白名单,只绑定这两个属性[Bind(Exclude="Id,Approved")]        //黑名单,不绑定这两个属性

  Bind特性可以应用于参数左侧也可以应用于实体Model类的顶部,应用于实体Modle的顶部则是对所有该实体绑定有效,而应用于参数左侧则只是对该action中的请求有效。

  如:

public ActionResult CommentAdd([Bind(Exclude="Approved")]Comment com) {   if (ModelState.IsValid)   {     //添加数据库     return Redirect("/Home/CommentManager");   }   else   {     return View(com);   } }

  另外,UpdateModel与TryUpdateModel也有一个重载版本来接收一个绑定列表:

UpdateModel(com, "", new string[] { "Id", "Name", "Content" });

  最后,还有一种就是视图模型,即另外在定义一个模型来专供视图使用,仅仅包括需要绑定的属性。

  另外,如果两个类有相同的Name属性,要同时绑定,区分HTML可以这样写:

客户名称:

销售员名称:

 

三、模型绑定原理

  在ASP.NET MVC中,用户请求道服务器的数据将被包装为Model数据对象,这个数据对象通常也被View用来提供显示的数据。在ASP.NET MVC中,提供了非常灵活的Model绑定机制,通过IModelBinder借口,定义了绑定Model数据的约定,并提供了一个接口的默认实现DefaultModelBinder。在大多数情况下,仅仅通过DefaultModelBinder就可以完成Model的绑定。

  如果需要的话,也可以自定义一个IModelBinder的实现,完成特定类型的Model绑定。

public interface IModelBinder{    object BindModel(ControllerContext controllerContext,ModelBindContext bindingContext);}

  1、绑定Model

  默认情况下,ASP.NET MVC使用DefaultModelBinder来绑定Model的数据。在传递Action参数的时候,ASP.NET MVC按照如下顺序查找匹配的数据:

  1. form表单中的数据;
  2. RouteData中的数据;
  3. QueryString中的数据;

  2、简单参数和复杂参数

  如果Action方法的参数类型是值类型和字符串类型,那么DefaultModelBinder将寻找与Action参数名称匹配的参数,如果没有对应的参数,那么Action的参数将试图赋予空引用。因此,对于简单类型的参数来说,参数的类型应该是可空的。

  多数情况下,我们会通过一个Model对象来处理复杂的参数,DefaultModelBinder会遍历Model对象的属性来绑定参数。

  如果不希望DefaultModelBinder对某个参数进行绑定,可以通过BindAttribute进行说明,其中定义了三个属性:

  • Include表示需要绑定的属性,各个属性之间以逗号进行分隔。
  • Exclude表示不需要绑定的属性,各个属性之前以逗号分隔。
  • Prefix表示请求参数的前缀。

  这些标签可以定义在Model上,说明在参数绑定过程中需要绑定的属性或者不需要绑定的属性,如:

[Bind(Include = "Name,Birthday")]public class Person{  public int Id { get; set; }  public string Name { get; set; }  public DateTime Birthday{ get; set; }}

   在UpdateModel方法中,指定包含的属性和不包含的属性。

UpdateModel(  person,     //Model  "person",    //Prefix  new[] { "Id","Name" },  //Include  new [] { "Birthday" }   //Exclude);

 

 

 

 

 

 

 

转载地址:http://bdlwa.baihongyu.com/

你可能感兴趣的文章
php中的短标签 太坑人了
查看>>
[译] 可维护的 ETL:使管道更容易支持和扩展的技巧
查看>>
### 继承 ###
查看>>
数组扩展方法之求和
查看>>
astah-professional-7_2_0安装
查看>>
函数是对象-有属性有方法
查看>>
uva 10107 - What is the Median?
查看>>
Linux下基本栈溢出攻击【转】
查看>>
c# 连等算式都在做什么
查看>>
使用c:forEach 控制5个换行
查看>>
java web轻量级开发面试教程摘录,java web面试技巧汇总,如何准备Spring MVC方面的面试...
查看>>
使用ansible工具部署ceph
查看>>
linux系列博文---->深入理解linux启动运行原理(一)
查看>>
Android反编译(一) 之反编译JAVA源码
查看>>
结合当前公司发展情况,技术团队情况,设计一个适合的技术团队绩效考核机制...
查看>>
python-45: opener 的使用
查看>>
cad图纸转换完成的pdf格式模糊应该如何操作?
查看>>
Struts2与Struts1区别
查看>>
网站内容禁止复制解决办法
查看>>
Qt多线程
查看>>