ASP.NET沒有魔法——ASP.NET MVC 模型綁定解析(上篇)


  前面文章介紹了ASP.NET MVC中的模型綁定和驗證功能,本着ASP.NET MVC沒有魔法的精神,本章內容將從代碼的角度對ASP.NET MVC如何完成模型的綁定和驗證進行分析,已了解其原理。

  本文的主要內容有:
  ● ModelBinder
  ● ValuePrivoder
  ● ModelMetadata
  ● 簡單模型與復雜模型
  ● 小結

ModelBinder

  ModelBinder是ASP. NET MVC用於模型綁定的核心組件,所有的ModelBinder都實現了IModelBinder接口,如下圖:

  

  該接口只有一個方法,那就是根據控制器以及綁定上下文完成模型綁定。
  在ASP.NET MVC中有不同的ModelBinder,它們分別用於綁定不同類型的數據,如普通的.Net對象、HTTP上傳的文件等。
  默認有以下5種ModelBinder:
  ● DefaultModelBinder:默認的模型綁定器,一般情況下從瀏覽器提交的請求都將使用默認處理器來綁定模型。
  ● HttpPostedFileBaseModelBinder:HTTP文件模型綁定
  ● ByteArrayModelBinder:綁定二進制數據。
  ● LinqBinaryModelBinder:將請求綁定到System.Data.Linq.Binary對象。參考: http://stephenwalther.com/archive/2009/02/25/asp-net-mvc-tip-49-use-the-linqbinarymodelbinder-in-your
  ● CancellationTokenModelBinder:提供了一個用於傳播模型綁定操作取消的機制。

  所有的ModelBinder都被一個名為ModelBinderDictionary的字典進行管理,而這個字典就存在於Controller類型的定義中,如下圖所示,它是一個被保護的內部屬性,用於Controller執行時完成模型綁定

  

  ModelBinderDictionary的定義:

  

  從ModelBinderDictionary的定義中可以看到它實現了以Type為Key、IModelBinder類型為值的字典接口以及集合接口,可以動態的根據ModelBinder的類型增減ModelBinder,除此之外還有一個DefaultBinder,在一般情況下其運行的實例如下:

  

  Controller中的ModelBinder字典包含了上面介紹的5個ModelBinder。更多關於自定義ModelBinder的內容可參考:https://www.cnblogs.com/Cwj-XFH/p/5977508.html

ValuePrivoder

  在前面的文章中介紹了ASP.NET MVC的模型綁定可以從Form Data、Query String以及Route Data等數據源中獲取數據,其原因是針對每一個數據源都有一個專門的數據提供器來獲取數據源的數據,在ASP.NET MVC中存在一個定義值提供器的接口IValueProvider:

  

  核心方法GetValue通過一個Key來獲取值,ContainsPrefix則判斷提供器所指的數據源中是否包含以指定字符串為前綴的Key。
  直接實現該接口的類型有3個:
  ● NameValueCollectionValueProvider:通過名稱和值共同存儲數據的集合,可以通過名稱查找到一個或多個值
  ● DictionaryValueProvider<TValue>:通過鍵值對存儲數據的泛型字典集合,字典的Key是唯一的,換句話說通過Key最多只能找到一個值
  ● ValueProviderCollection:一個特殊的值提供器,它包含了所有相關的值提供器,在模型綁定中就是這個列表通過遍歷的方式,調用相關值提供器的獲取值方法來完成數據獲取的。

  之前說過針對不同的數據源都有一個特定值提供器,那么這些提供器是如何實現的呢?它們是根據特性分別繼承NameValueCollectionValueProvider及DictionaryValueProvider<TValue>類型來實現的,其具體分類如下,一共有7種不同數據源的值提供器:
  ● NameValueCollectionValueProvider:
    ○ JQueryFormValueProvider:用於獲取被Jquery格式化的Form值。
    ○ FormValueProvider:用於獲取Form表單的值。
    ○ QueryStringValueProvider:用於獲取查詢字符串的值。
  ● DictionaryValueProvider<TValue>
    ○ ChildActionValueProvider:用於獲取子Action方法的值。
    ○ JsonValueProvider:用於獲取請求中以Json格式傳輸的值(注:沒有該類型的值提供器,Json的值提供器直接由JsonValueProviderFactory創建一個DictionaryValueProvider<object>類型的字典值提供器)。
    ○ HttpFileCollectionValueProvider:用於從Http請求中的文件集合中獲取文件數據。
    ○ RouteDataValueProvider:用於從Route Data中獲取值
  所有的值提供器都是由對應的工廠創建的,在默認情況下ASP.NET MVC中存在以下7種值提供器工廠,剛好對應上面的7種值提供器,其實現接口定義如下:

  

  每一個工廠的GetValueProvider方法獲取對應的值提供器。
  所有的提供器工廠在MVC中被一個名為ValueProviderFactories的類型維護,該類型以硬編碼的方式維護了一個靜態、只讀的提供器工廠列表:

  

   運行時結果,一共有7個工廠:

  

  下圖是ASP.NET MVC在未特殊配置的情況下Controller的運行狀態:

  

 下圖是發送Json格式Post請求的狀態,ValueProvider中多了一個用來獲取Json數據的字典值提供器:

 發起請求的內容:

  

  請求中的值提供器與之前的相比多了一個用於提供Json數據的DictionaryValueProvider<object>類型:

  

ModelMetadata

  Metadata譯為元數據,是一種描述數據的數據,而這里加上了Model,那么就是說描述Model數據的數據,下圖為ASP.NET MVC中的一個Model定義:

   

  從圖中代碼用語言可以這樣描述:
  ● 該對象有3個屬性。
  ● 其中UserName是String類型的,必填並且格式為Email格式,展示名稱為用戶名。
  ● Password以及ConfirmPassword都是String類型,且類型都為密碼,它們除了展示名稱不同外,ConfirmPassword還需要和Password相比較,如果不同則給出相應的錯誤提示。

  而在MVC里面是通過ModelMatedata來對Model進行描述的,先看一下ASP.NET MVC中的ModelMetadata類型:

  

    

  從中可以看到一些是否只讀、是否必填、模型類型、屬性(同樣是ModelMetadata類型)、展示名稱、是否是復雜類型等描述信息。換句話就是ASP.NET MVC通過ModelMetadata可以對模型的屬性是否只讀、是否必填、類型等相應信息進行描述,甚至還包含了模型驗證器來完成合法性驗證。
  這里需要注意的是模型類型本身通過一個ModelMetadata來描述,而類型的屬性同樣被ModelMetadata描述,就是說ModelMetadata描述類型的結構是與對應類結構有相同的層次。

  注:ModelMetadata涉及到View的渲染,關於View的內容會在后續文章中介紹。

簡單模型與復雜模型

  在ModelMetadata類型中有一個名為IsComplexType的屬性,用於表示該類型是否為復雜類型,那么什么是復雜類型?相對應的什么是簡單類型?

  

  上圖是IsComplexType的實現代碼,其核心有兩個點:
  1. 通過當前模型類型來獲取一個轉換器(注:TypeDescriptor是一個用來獲取類型相關信息的類型,如特性、屬性、事件等,當然也包括類型轉換器,下圖是TypeDescriptor部分定義)。

  

  

  2.  獲取到類型轉換器后,通過轉換器判斷該類型是否能夠從字符串轉換,如果能那么它就是簡單類型否則為復雜類型。(下圖為TypeConverter的部分定義)

  

  知道了簡單類型與復雜類型的區別,那么它們在MVC中有什么意義呢?
  首先對於ASP.NET MVC來說,它接收到的Http請求無論Header、Body等它實質上都是字符串,那么根據Http協議從中取出來的數據也是字符串,但是對於MVC的Action方法來說,它接受的參數可能是字符串的,也可能是數字、時間等其它類型,所以這里需要一個從字符串到其它類型的轉換過程,而簡單類型的目的就在於MVC進行模型綁定時可以直接根據名稱從ValueProvider中獲取到值(一個字符串),然后通過類型轉換器將該字符串轉換成所需類型
  下面就介紹一些“理所當然”的類型轉換器:
  ● 數字:下圖是數字轉換器的基類,它的CanConvertFrom方法的實現直接硬編碼了能夠從string類型轉換數字類型(不同數字類型有具體的實現,這里不再介紹)。

   

    ● 時間:同樣的時間類型也能從字符串轉換。

  

    ● 布爾:布爾類型能夠從字符串轉換。

  

  為什么說“理所當然”?因為在實際的開發中某action需要一個時間參數,那么在表單中填寫或者通過一些js組件選擇一個日期,然后提交到服務器這個填寫的時間“就是”一個時間類型,填寫的數字也就是數字類型,一切都是理所當然的。但是打着沒有“魔法”的目的,需要知道的是,哪怕最簡單的布爾類型實際上也進行了轉換工作,下圖就是Boolean轉換器的轉換代碼:

  

  在MVC中簡單模型和復雜模型綁定的方式是不同的,也很容易理解簡單模型僅需要一個字符串就可以完成轉換,而復雜模型還需要更多的操作。
  在ASP.NET MVC中還有一種用法,就是自定義將特殊格式的字符串轉換成特定對象,常用的例子就是坐標(經緯度)的轉換,下面將使用逗號分隔字符串的格式定義用戶注冊的RegisterViewModel:
  1、創建一個RegisterViewModel的轉換器,繼承TypeConverter類型並重載CanConvertFrom及ConvertFrom方法即可:

  

  2、使用TypeConverter特性在RegisterViewModel類型上使用該轉換器:

  

  3、通過Postman模擬請求:

   

  注:model為action參數名稱。
  Action能夠正確的獲取到數據:

  

小結

  本篇內容主要是介紹了ASP.NET MVC中模型綁定的主要組件與概念,ValueProvider提供數據、ModelMetadata描述數據、ModelBinder綁定數據,綁定數據過程中不可或缺的數據驗證、轉換功能分別對應了ModelValidation以及TypeConverter類型。下一篇文章將介紹ASP.NET MVC在Controller執行時如何結合這些組件實現模型綁定的邏輯。

PS.由於篇幅較長所以將模型綁定解析的內容分為兩篇,下篇將盡快整理並發出。祝各位元宵快樂(*^_^*)

參考:
  http://stephenwalther.com/archive/2009/02/25/asp-net-mvc-tip-49-use-the-linqbinarymodelbinder-in-your
  https://www.cnblogs.com/Cwj-XFH/p/5977508.html

本文鏈接:http://www.cnblogs.com/selimsong/p/8484482.html 

ASP.NET沒有魔法——目錄


免責聲明!

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



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