ASP.NET MVC Model元數據(二)

ASP.NET MVC Model元數據(二)

前言

在上篇中,給大家留個對Model元數據的印象,並沒有對Model元數據有過多的講解,而在本篇中也不會對Model元數據的本身來解釋,而是針對於它的生成過程,讓大家能夠清楚的瞭解到系統框架是在什麼時候生成Model元數據的,對於Model元數據生成篇幅初定爲兩篇,本篇爲它的整體的生成過程,下篇則爲詳細的生成過程並且會對它本身做一個粗略的介紹,希望大家看完能夠有所收穫

 

Model元數據

  • 什麼是Model元數據?
  • 生成Model元數據的過程【一】
  • 生成Model元數據的過程【二】
  • ModelMetaData的定義、詳解
  • Model元數據應用(常用特性應用)-1
  • Model元數據應用(自定義視圖模板)-2
  • Model元數據應用(IMetadataAware接口使用)-3

 

生成Model元數據的過程【一】

還是如前篇說的那樣,既然叫Model元數據(Model指的是視圖模型)那麼肯定跟Model有關係了,而在我們MVC項目中一般是什麼時候會對Model進行操作呢?一般情況下都是在通過控制器的行爲請求一個視圖的時候,而控制器行爲的參數即爲Model,然後在行爲方法中做一些處理然後再傳遞給視圖。然後再根據上篇最後的一個示意圖來看,

圖1

 

生成Model元數據的地方已經鎖定到了行爲方法,想象一下肯定是不可能在行爲方法中來生成的,因爲那是我們自定義邏輯的地方。那是在什麼地方呢?

想必大家看過之前的對過濾器篇幅的介紹,在ASP.NET MVC 過濾器(三)中對行爲過濾器的執行過程講解的時候,中間有提到過模型綁定器,並且說到了系統框架所要使用到的自定義模型綁定器,而使用這個自定義模型綁定器所需要的兩個參數是非常重要的,一個是表示當前控制器上下文的對象ControllerContext,另一個則是生成Model元數據的關鍵,也是調用自定義模型綁定器的關鍵參數ModelBindingContext類型。看下

圖2

 

而在ModelBindingContext類型中有個重要的屬性,即爲Model元數據類型ModelMetadata,由此可以知道在我們的控制器行爲執行之前,對應控制器行爲的Model的Model元數據ModelMetadata類型已經生成了。(這部分內容詳見過濾器篇幅)

而它是怎麼生成的呢?是通過系統框架中默認提供的提供程序來生成的,是哪些個類型呢?

圖3

那我們就先看一下最頂層的基類ModelMetadataProvider的定義:

代碼1-1

public abstract class ModelMetadataProvider
    {
        // 摘要:
        //     在派生類中重寫時,初始化派生自 System.Web.Mvc.ModelMetadataProvider 類的對象的新實例。
        protected ModelMetadataProvider();
        public abstract IEnumerable<ModelMetadata> GetMetadataForProperties(object container, Type containerType);
        public abstract ModelMetadata GetMetadataForProperty(Func<object> modelAccessor, Type containerType, string propertyName);
        public abstract ModelMetadata GetMetadataForType(Func<object> modelAccessor, Type modelType);
    }

 

非常明白的定義,三個抽象方法,這裏我們只需先看GetMetadataForType()方法,其它兩個暫時不管下篇中會有講到,因爲先看GetMetadataForType()方法呢?因爲它是生成ModelMetadata類型的入口,第一個參數暫時忽略,第二個參數嘛很重要了,是ParameterDescriptor類型的ParameterType屬性,表示着Model的類型(也就是控制器方法參數的Type類型),現在我們來看下圖4

圖4

 

圖4中藍色線條爲主要流程,紅色線條是在藍色處理之後執行的流程。

上面說到,入口方法是爲抽象方法,那是怎麼具體實現的呢,從圖4中可以看到是由實現了ModelMetadataProvider的類型AssociatedMetadataProvider類型來進行處理的,從圖4可以看到首先是獲取一個AttributeList的類型,AttributeList類型表示着從AssociatedMetadataProvider類型GetMetadataForType()方法參數modelType類型上的特性集合,對了AssociatedMetadataProvider類型是比較重要的類型,我們先來看一下它的定義:

public abstract class AssociatedMetadataProvider : ModelMetadataProvider
    {
        protected AssociatedMetadataProvider();
        protected abstract ModelMetadata CreateMetadata(IEnumerable<Attribute> attributes, Type containerType, Func<object> modelAccessor, Type modelType, string propertyName);
        protected virtual IEnumerable<Attribute> FilterAttributes(Type containerType, PropertyDescriptor propertyDescriptor, IEnumerable<Attribute> attributes);
        public override IEnumerable<ModelMetadata> GetMetadataForProperties(object container, Type containerType);
        protected virtual ModelMetadata GetMetadataForProperty(Func<object> modelAccessor, Type containerType, PropertyDescriptor propertyDescriptor);
        public override ModelMetadata GetMetadataForProperty(Func<object> modelAccessor, Type containerType, string propertyName);
        public override ModelMetadata GetMetadataForType(Func<object> modelAccessor, Type modelType);
        protected virtual ICustomTypeDescriptor GetTypeDescriptor(Type type);
    }

方法有點多,暫時不用管,大多數方法都是用來在遞歸生成Model元數據的時候使用的(具體的過程會在下篇中講解)。好了切回主題接着上面的內容來說,AttributeList類型的由來,是通過ModelMetadataProvider的GetTypeDescriptor()方法根據Model的類型(這裏暫且先這麼理解,等看完下個篇幅就會知道這裏也有可能是Model中的屬性類型)來生成一個ICustomTypeDescriptor類型(可以想象成這是對於一種對象類型元數據描述對象的抽象定義。讀起來有點繞口,但是確實是這麼個意思)。而系統會有個默認的自定義實現來實現這個接口類型,我們通過這個默認的實現來獲得Model類型的AttributeList類型。

在有了AttributeList類型後,我們就可以調用AssociatedMetadataProvider類型的CreateMetadata()方法來創建Model元數據對象,但是這個CreateMetadata()的定義是抽象的,而真正的實現是由繼承了AssociatedMetadataProvider類型的DataAnnotationsModelMetadataProvider類型,由此過後我們生成得到ModelMetadata元數據對象(真正的過程比較繁瑣,不然也不會另起一篇專門用來講解生成的過程),得到了Model元數據對象過後並沒有結束,而是繼續調用了AssociatedMetadataProvider類型的ApplyMetadataAwareAttributes()方法,並在此方法中,系統框架會調用我們自定義實現了IMetadataAware接口類型的對象,來對Model元數據對象進行個性化修改,並且最後纔會真正的返回Model元數據對象。

有可能看到這裏有的朋友對Model元數據還是不怎麼清楚和了解,朋友們急我也急,如果我分享的這些知識大家看完都不知所云那我又是何必呢。先不要急看了多少就是多少在看完這個Model元數據系列的文章後應該會有所瞭解,將在後續的篇章中慢慢的揭開它的祕密。謝謝大家的支持。

 

 

 

 

作者:金源

發佈了74 篇原創文章 · 獲贊 1 · 訪問量 8萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章