ASP.NET MVC——模型綁定

来源:http://www.cnblogs.com/skyshalo/archive/2016/12/27/6224018.html
-Advertisement-
Play Games

這篇文章我們來講講模型綁定(Model Binding),其實在初步瞭解ASP.NET MVC之後,大家可能都會產生一個疑問,為什麼URL片段最後會轉換為例如int型或者其他類型的參數呢?這裡就不得不說模型綁定了。模型綁定是指,用瀏覽器以HTTP請求方式發送的數據來創建.NET對象的過程。每當定義具 ...


   這篇文章我們來講講模型綁定(Model Binding),其實在初步瞭解ASP.NET MVC之後,大家可能都會產生一個疑問,為什麼URL片段最後會轉換為例如int型或者其他類型的參數呢?這裡就不得不說模型綁定了。模型綁定是指,用瀏覽器以HTTP請求方式發送的數據來創建.NET對象的過程。每當定義具有參數的動作方法時,一直是在依賴著這種模型綁定過程。

準備項目

    我們先來創建一個MVC項目,名叫MVCModels,併在Models文件夾中創建一個新的類文件Person。

 1 using System;
 2 
 3 namespace MVCModels.Models
 4 {
 5     public class Person
 6     {
 7         public int PersonId { get; set; }
 8         public string FirstName { get; set; }
 9         public string LastName { get; set; }
10         public DateTime BirthDate { get; set; }
11         public Address HomeAddress { get; set; }
12         public Role Role { get; set; }
13     }
14 
15     public class Address
16     {
17         public string Line { get; set; }
18         public string City { get; set; }
19         public string PostalCode { get; set; }
20         public string Country { get; set; }
21     }
22 
23     public enum Role
24     {
25         Admin,
26         User,
27         Guest
28     }
29 }

    另外定義一個Home控制器。

 1 using MVCModels.Models;
 2 using System.Linq;
 3 using System.Web.Mvc;
 4 
 5 namespace MVCModels.Controllers
 6 {
 7     public class HomeController : Controller
 8     {
 9         private Person[] personDate = {
10             new Person { PersonId = 1, FirstName = "Adam", LastName = "Freeman", Role = Role.Admin },
11             new Person { PersonId = 2, FirstName = "Jacqui", LastName = "Griffyth", Role = Role.User },
12             new Person { PersonId = 1, FirstName = "John", LastName = "Smith", Role = Role.Guest },
13         };
14         public ActionResult Index(int id)
15         {
16             Person dataItem = personDate.Where(p => p.PersonId == id).First();
17             return View(dataItem);
18         }
19     }
20 }

    接下來再創建一個視圖文件Index。

@model MVCModels.Models.Person
@{
    ViewBag.Title = "Index";
    Layout = "~/Views/Shared/_Layout.cshtml";
}

<h2>Person</h2>
<div>
    <label>ID:</label>
    @Html.DisplayFor(m => m.PersonId)
</div>
<div>
    <label>First Name:</label>
    @Html.DisplayFor(m => m.FirstName)
</div>
<div>
    <label>Last Name:</label>
    @Html.DisplayFor(m => m.LastName)
</div>
<div>
    <label>Role:</label>
    @Html.DisplayFor(m => m.Role)
</div>

理解模型綁定

    模型綁定是HTTP請求與C#方法之間的一個橋梁,它根據 Action 方法中的 Model 類型創建 .NET 對象,並將 HTTP 請求數據經過轉換賦給該對象。當我們啟動項目,並導航到/Home/Index/1,我們會看見圖下:

    當我們請求 /Home/Index/1 URL 時,路由系統便將最後一個片段值 1 賦給了 id 變數。action invoker 通過路由信息知道當前的請求需要 Index action 方法來處理,但它調用 Index action 方法之前必須先拿到該方法參數的值。 預設的動作調用器ControllerActionInvoker,要依靠模型綁定器來生成調用動作所需要的的數據對象,模型綁定器由IModelBinder介面所定義。在本例中,動作調用器會檢查Index方法,併發現它具有一個int型參數,於是會查找負責int值綁定的綁定器,並調用它的BindModel方法。

使用預設模型綁定器

    雖然應用程式可以自定義模型綁定器,但是大多數情況下還是依靠內建的綁定器類DefaultModelBinder。當動作調用器找不到綁定某個類型的自定義綁定器時,就會使用這個預設的模型綁定器,預設情況下,這個模型綁定器會搜索四個位置。如下表所示:

DefaultModelBinder類查找參數數據的順序
描述
Request.From 由用戶在HTML的表單元素中提供值
RouteData.Values 用應用程式路由獲得的值
Request.QueryString 包含在請求URL中的查詢字元串部分的數據
Request.Files 請求中上傳的文件

    這些位置將被依次搜索,在本例中,DefaultModelBinder會為id參數查找以下的一個值:

    1.Request.Form["id"]

    2.RouteData.Values["id"]

    3.Request.QueryString["id"]

    4.Request.Files["id"]

綁定簡單類型

    處理簡單參數類型時,DefaultModelBinder會使用System.ComponentModel.TypeDescriptor類,講請求數據獲得的字元參數值轉換為參數類型,如果無法轉換,那麼DefaultModelBinder便不能綁定到Model。比如訪問/Home/Index/apple,便會出現如圖所示:

    預設模型綁定器看到需要的是int值,如果視圖將URL中提供的apple值轉換為int型,就會出錯,此時我們可以修改代碼,為參數提供一個預設值,這樣當預設綁定器無法轉換時也不會出錯了。

1 ...  
2 public ActionResult Index(int id = 1)
3 {
4     Person dataItem = personDate.Where(p => p.PersonId == id).First();
5     return View(dataItem);
6 }

綁定複雜類型

     當動作方法參數是符合類型時(即不能用TypeConverter類進行轉換的屬性),DefaultModelBinder類將用反射類獲取public屬性集。好的,我們現在來修改代碼。

 1 using MVCModels.Models;
 2 using System.Linq;
 3 using System.Web.Mvc;
 4 
 5 namespace MVCModels.Controllers
 6 {
 7     public class HomeController : Controller
 8     {
 9         private Person[] personDate = {
10             new Person { PersonId = 1, FirstName = "Adam", LastName = "Freeman", Role = Role.Admin },
11             new Person { PersonId = 2, FirstName = "Jacqui", LastName = "Griffyth", Role = Role.User },
12             new Person { PersonId = 1, FirstName = "John", LastName = "Smith", Role = Role.Guest },
13         };
14         public ActionResult Index(int id = 1)
15         {
16             Person dataItem = personDate.Where(p => p.PersonId == id).First();
17             return View(dataItem);
18         }
19         public ActionResult CreatePerson()
20         {
21             return View(new Person());
22         }
23         [HttpPost]
24         public ActionResult CreatePerson(Person model)
25         {
26             return View("Index", model);
27         }
28     }
29 }

    另外創建一個CreatePerson視圖。

 1 @model MVCModels.Models.Person
 2 @{
 3     ViewBag.Title = "CreatePerson";
 4     Layout = "~/Views/Shared/_Layout.cshtml";
 5 }
 6 
 7 <h2>CreatePerson</h2>
 8 @using (Html.BeginForm())
 9 {
10     <div>
11         <label>
12             @Html.LabelFor(m => m.PersonId)
13             @Html.EditorFor(m => m.PersonId)
14         </label>
15     </div>
16     <div>
17         <label>
18             @Html.LabelFor(m => m.FirstName)
19             @Html.EditorFor(m => m.FirstName)
20         </label>
21     </div>
22     <div>
23         <label>
24             @Html.LabelFor(m => m.LastName)
25             @Html.EditorFor(m => m.LastName)
26         </label>
27     </div>
28     <div>
29         <label>
30             @Html.LabelFor(m => m.Role)
31             @Html.EditorFor(m => m.Role)
32         </label>
33     </div>
34     <button type="submit">Submit</button>
35 }

    在表單回遞給CreatePerson方法時,預設綁定器發現動作方法需要一個Person對象,便會依次處理每個屬性。綁定器會從請求中找到每一個值。如果一個屬性需要另一個複合類型時,那麼該過程會重覆執行。例如本例中,Person類的HomeAddress屬性是Address類型,在為Line屬性查找值時,模型綁定器查找的是HomeAddress.Line的值,即模型對象的屬性名(HomeAddress)與屬性類型(Address)的屬性名(Line)的組合。

Bind特性

    我們還可以用bind特性為 Address 類型的參數綁定 Person 對象中的 HomeAddress 屬性值,例如這樣:

 

1 public ActionResult DisplayAddress([Bind(Prefix="HomeAddress")]Address address) 
2 {
3     return View(address);
4 }

 

    DisplayAddress action 方法的參數類型 Address 不一定必須是 Person 的 HomeAddress 屬性的類型,它可以是其他類型,只要該類型中含有City

或 Country 同名的屬性就都會被綁定到。不過,要註意的是,使用 Bind 特性指定了首碼後,需要提交的表單元素的 name 屬性必須有該首碼才能被綁定。Bind 特性還有兩個屬性,Exclude 和 Include。它們可以指定在 Mdoel 的屬性中,Binder 不查找或只查找某個屬性,即在查找時要麼只包含這個屬性要麼不包含這個屬性。如下麵的 action 方法:

1 public ActionResult DisplayAddress([Bind(Prefix = "HomeAddress", Exclude = "Country")]Address address) 
2 {
3     return View(address);
4 }

這時 Binder 在綁定時不會對 Address 這個 Model 的 Country 屬性綁定值。

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • rotatelogs 是 Apache 2.2 中自帶的管道日誌程式,參數如下(參見:http://lamp.linux.gov.cn/Apache/ApacheMenu/programs/rotatelogs.html) 語法 rotatelogs [ l ] logfile [ rotation ...
  • 步驟:1. 掛在光碟 配置本地源; 2. 安裝redhat_lsb; 3. 安裝chrome 1 mount /dev/cdrom /mnt cd /etc/yum.repos.d/ vi local.repo (這裡要添加內容的: [local_cd] gpgcheck=0 enable=1 ba ...
  • 右擊開始->屬性->開始菜單->自定義>點擊運行命令(選擇)->確定 ...
  • Sciter中的Tiscript腳本不是標準的Javascript,是對Javascript的擴展。所以在常用的編輯器和IDE上對於高亮的支持很不好。 不過在Sciter論壇中找到了在VSCode上的高亮支持。安裝方法比較簡單。 首先點此下載插件。 之後在Sciter插件所在目錄執行 然後在VSCo ...
  • 最近修改的一個需求要求捕獲鍵盤輸入的 Text,包括各種標點符合。 最開始想到的是 PreviewKeyDown 或者 PreviewKeyUp 這樣的鍵盤事件。 但是這兩個事件的對象 KeyEventArgs 不夠用,得依靠判斷 Key 來判斷輸入了什麼,然後再根據判斷寫死一個相應鍵位的 Text ...
  • 主要是基於達索軟體Composer Player、的基礎上做些二次開發。 public class ComposerToolBarSetting { public bool AntiAliasingOnIdle { get; set; }//抗鋸齒化 public bool AssySelectio ...
  • ...
  • 最近需要用到多選框,Asp.Net MVC自帶的@Html.ListBox或@Html.ListBoxFor的顯示效果太差,於是找到了Select2進行優化,並正式瞭解了多選框的操作方法。 首先介紹多選框的操作方法; 一、定義一個MultipleSelectModel 二、在Controller中實 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...