ASP.NET Web API Model-ValueProvider

ASP.NET Web API Model-ValueProvider

前言

前面一篇講解了Model元數據,Model元數據是在Model綁定中很重要的一部分,只是Model綁定中涉及的知識點比較多,對於ASP.NET MVC框架來說ASP.NET Web API框架中在Model綁定部分又新增了參數綁定這麼一個機制,這些內容都會在後面的篇幅中說明,前面的這些篇幅都是講解理論上的知識也沒有涉及到示例的演示,這個大家不用急在最後Model部分的基礎知識講解完之後我會把前面所講的全部串聯起來,而今天這個篇幅給大家帶來的就是在Model綁定中起到至關重要的一個環節,大家這個不用去管什麼Model綁定,而是單純的去了解ValueProvider這一系列的對象模型,因爲在沒有結合前後知識點串聯起來之前,我們都不知道ValueProvider是在哪執行的。所以我們還是老老實實瞭解一下ValueProvider相關的對象吧。

 

Model-ValueProvider

1

wKioL1QZfzPR_T_QAAGM5t3jfmM887.jpg


IValueProvider接口類型--ValueProvider行爲約束

首先我們看一下圖1中右邊的部分,起頭的就是一個IValueProvider接口類型,我們就來看一下接口的定義:

示例代碼1-1

   publicinterfaceIValueProvider
   {
        boolContainsPrefix(stringprefix);
        ValueProviderResultGetValue(stringkey);
   }

我們在代碼1-1中看到,IValueProvider接口中定義了兩個方法,一個是ContainsPrefix()方法,接收string類型的參數並且返回的是bool值類型,這個方法表示的就是根據指定的前綴值查看當前的ValueProvider中是否存在這個前綴值,這個下面會有示例稍後再說,然後就是GetValue()方法,是根據執行的鍵值返回當中的對應的值,從這裏一看我們就大概能猜到這個ValueProvider應該是類似於鍵值隊一樣的類型,而返回的結果被封裝在了ValueProviderResult類型當中,這個類型稍後會有說明。在IValueProvider接口約束好ValueProvider值提供程序的行爲後,我們應該就來看一看ValueProvider值提供程序的基礎結構了。不過呢在此之前我們還是要根據圖1中所示的那樣,先來看一下IEnumerableValueProvider接口類型的定義,這個接口主要負責什麼呢?


IEnumerableValueProvider接口類型-ValueProvider行爲約束

示例代碼1-2

   publicinterfaceIEnumerableValueProvider : IValueProvider
   {
        IDictionary<string, string>GetKeysFromPrefix(stringprefix);
   }

從代碼1-2中可以很清楚明瞭的看到IEnumerableValueProvider接口類型的職責很簡單,就是對指定的前綴值進行檢索,並且最後以鍵值隊的形式返回,這個在下面會有示例詳細說明。


NameValuePairsValueProvider類型-ValueProvider基礎結構

示例代碼1-3

   publicclassNameValuePairsValueProvider : IEnumerableValueProvider, IValueProvider
   {
        publicNameValuePairsValueProvider(Func<IEnumerable<KeyValuePair<string, string>>>valuesFactory, CultureInfoculture);
        publicNameValuePairsValueProvider(IEnumerable<KeyValuePair<string, string>>values, CultureInfoculture);
 
        publicvirtualboolContainsPrefix(stringprefix);
        publicvirtualIDictionary<string, string>GetKeysFromPrefix(stringprefix);
        publicvirtualValueProviderResultGetValue(stringkey);
    }

在代碼1-3中我們看到NameValuePairsValueProvider類型的定義,首先就說說它的構造函數吧,兩個構造函數的區別在於第一個是Func<IEnumerable<KeyValuePair<string, string>>>類型的構造函數參數,第二個是IEnumerable<KeyValuePair<string, string>>類型的構造函數參數,第二個構造函數的首個參數類型實際就是第一個構造函數首個參數的返回類型,這裏大家都看得到,其實在內部實現,也是這樣的聲明第一個構造函數是沒什麼的,聲明第二個構造函數的時候其實就把參數再次封裝爲委託。

對於KeyValuePair<T,U>類型可以理解爲鍵值隊的子項,在它的類型中只有一個鍵值對應一個值只有一項就是它本身。

至於剩下的三個方法我們還是靠簡單的示例來說明一下。

示例代碼1-4

    publicclassValueProviderCaseController : ApiController
   {
        publicstringGet()
        {
            
            KeyValuePair<string,string>[] dictionary=newKeyValuePair<string,string>[]
            {
                newKeyValuePair<string,string>("EmployeesInfo.Name","Jinyuan"),
                newKeyValuePair<string,string>("EmployeesInfo.Age","24"),
                newKeyValuePair<string,string>("EmployeesInfo.Sex","男"),
                newKeyValuePair<string,string>("EmployeesInfo.AddressInfo.AddressInfo","南京市"),
                newKeyValuePair<string,string>("EmployeesInfo.AddressInfo.ZipCode","210000")
            };
            NameValuePairsValueProvidernameValuePairsValueProvider=newNameValuePairsValueProvider(dictionary,null);
            StringBuilderstrBuilder=newStringBuilder();
            NameValuePairsPrefixAnalysis(strBuilder, nameValuePairsValueProvider, "EmployeesInfo");
            returnstrBuilder.ToString();
        }
 
        privatevoidNameValuePairsPrefixAnalysis(StringBuilderstringbuilder, NameValuePairsValueProvidernamevaluepairs,stringprefix)
        {
            IDictionary<string, string>dictionarys=namevaluepairs.GetKeysFromPrefix(prefix);
            
            if (dictionarys.Count>0)
            {
                Console.WriteLine(prefix+"爲前綴的數據源Key值檢索……");
                foreach (varitemindictionarys)
                {
                    Console.WriteLine("Key:"+item.Key+" Value:"+item.Value);
                }
                foreach (KeyValuePair<string, string>keyvalueindictionarys)
                {
                    NameValuePairsPrefixAnalysis(stringbuilder, namevaluepairs, keyvalue.Value);
                }
            }
            else
            {
                stringbuilder.AppendLine(prefix+":"+namevaluepairs.GetValue(prefix).RawValue.ToString());
            }
        }
   }

我們來看代碼1-4,首先我在Get()方法中定義了一個KeyValuePair<string,string>[]類型,爲了能夠實例化NameValuePairsValueProvider類型,在此之後大家可以看到我調用了一個我自定義的NameValuePairsPrefixAnalysis()方法,並且在其中使用NameValuePairsValueProvider類型的實例調用了GetKeysFromPrefix()方法,也就是代碼1-2所約束的那個行爲。這個時候我們先來看一下表1.

 

1



Key

Value

EmployeesInfo.Name

Jinyuan

EmployeesInfo.Age

24

EmployeesInfo.Sex

EmployeesInfo.AddressInfo.AddressInfo

南京市

EmployeesInfo.AddressInfo.ZipCode

210000


1所表示的就是初始數據源,也就是我們定義的KeyValuePair<string,string>[]類型的鍵、值示意表。

然而在我們使用NameValuePairsValueProvider類型的實例已” EmployeesInfo”作爲前綴調用了GetKeysFromPrefix()方法後返回的IDictionary<string,string>類型的值如表2.

 

2


Key

Value

Name

EmployeesInfo.Name

Age

EmployeesInfo.Age

Sex

EmployeesInfo.Sex

AddressInfo

EmployeesInfo.AddressInfo

這裏的表2值只是第一層的關係值。

在此之後我們輸出的當前的所要檢索的前綴值以及檢索過後的值,並且會遍歷表2裏的Value值作爲前綴值再次的對數據源進行前綴檢索,如果沒有了則說明已經沒有可檢索的了。

並且使用NameValuePairsValueProvider類型實例調用的GetValue()方法根據最後已經檢索不出來有後綴的前綴值,也就是原始數據源當中的Key值了。

最後我們看一下結果。

2

wKiom1QZgH3SDw29AAL_-3eRrtA219.jpg

在客戶端我們獲取到了值,當然這裏只是演示示例,值提供程序提供的值方向反了。

然後我們可以在服務端看到檢索的記錄,可以明確的看到有兩層的結構在其中。有興趣的朋友深入一下看下檢索的具體實現方式。

 

QueryStringValueProvider類型-ValueProvider特定結構

示例代碼1-5

   publicclassQueryStringValueProvider : NameValuePairsValueProvider
   {
        publicQueryStringValueProvider(HttpActionContextactionContext, CultureInfoculture)
            : base(func, culture)
        {
            Func<IEnumerable<KeyValuePair<string, string>>>func=null;
            if (func==null)
            {
                func= () =>actionContext.ControllerContext.Request.GetQueryNameValuePairs();
            }
        }
   }

從代碼1-5中可以看到在QueryStringValueProvider類型初始化的時候構造函數中的執行,把從請求查詢字符串作爲原始數據來源封裝爲委託類型,然後調用基類的構造函數。


RouteDataValueProvider類型-ValueProvider特定結構

示例代碼1-6

   publiccla***outeDataValueProvider : NameValuePairsValueProvider
   {
        publicRouteDataValueProvider(HttpActionContextactionContext, CultureInfoculture)
            : base(func, culture)
        {
            Func<IEnumerable<KeyValuePair<string, string>>>func=null;
            if (func==null)
            {
                func= () =>GetRoutes(actionContext.ControllerContext.RouteData);
            }
        }
   }

同上面1-5一樣的道理,這裏使用了HttpRouteData中的Values值作爲原始數據源。

 

還有圖1中的左邊部分會在後面的篇幅中講解,在這裏講不適合,會感覺不連貫,雖然看完本篇不知道這個起到了什麼作用也不知道怎麼去使用,感覺被掐住了脖子一樣的難受,但是後面我會把前面所講的內容全部串聯起來做一個演示,所示這是基礎部分的知識,就是一個鋪墊。


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章