激情久久久_欧美视频区_成人av免费_不卡视频一二三区_欧美精品在欧美一区二区少妇_欧美一区二区三区的

服務(wù)器之家:專注于服務(wù)器技術(shù)及軟件下載分享
分類導(dǎo)航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術(shù)|正則表達(dá)式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務(wù)器之家 - 編程語言 - ASP.NET教程 - asp.net mvc-Controllerl篇 ControllerDescriptor

asp.net mvc-Controllerl篇 ControllerDescriptor

2019-10-10 10:57asp.net技術(shù)網(wǎng) ASP.NET教程

asp.net mvc-Controllerl篇 ControllerDescriptor

現(xiàn)在我們首先來看看ActionInvoker屬性的定義吧: 

復(fù)制代碼代碼如下:


public IActionInvoker ActionInvoker { 
get { 
if (_actionInvoker == null) { 
_actionInvoker = CreateActionInvoker(); 

return _actionInvoker; 

set { 
_actionInvoker = value; 


protected virtual IActionInvoker CreateActionInvoker() { 
return new ControllerActionInvoker(); 


和TempDataProvider屬性定義一樣,大家一定要習(xí)慣這些代碼啊。 
而ControllerActionInvoker的定義也很簡單,但是這個類卻不簡單啊。 
讓我們來看看你InvokeAction的定義吧: 

復(fù)制代碼代碼如下:


public virtual bool InvokeAction(ControllerContext controllerContext, string actionName) { 
if (controllerContext == null) { 
throw new ArgumentNullException("controllerContext"); 

if (String.IsNullOrEmpty(actionName)) { 
throw new ArgumentException(MvcResources.Common_NullOrEmpty, "actionName"); 


ControllerDescriptor controllerDescriptor = GetControllerDescriptor(controllerContext); 
ActionDescriptor actionDescriptor = FindAction(controllerContext, controllerDescriptor, actionName); 
if (actionDescriptor != null) { 
FilterInfo filterInfo = GetFilters(controllerContext, actionDescriptor); 

try { 
AuthorizationContext authContext = InvokeAuthorizationFilters(controllerContext, filterInfo.AuthorizationFilters, actionDescriptor); 
if (authContext.Result != null) { 
// the auth filter signaled that we should let it short-circuit the request 
InvokeActionResult(controllerContext, authContext.Result); 

else { 
if (controllerContext.Controller.ValidateRequest) { 
ValidateRequest(controllerContext); 


IDictionary<string, object> parameters = GetParameterValues(controllerContext, actionDescriptor); 
ActionExecutedContext postActionContext = InvokeActionMethodWithFilters(controllerContext, filterInfo.ActionFilters, actionDescriptor, parameters); 
InvokeActionResultWithFilters(controllerContext, filterInfo.ResultFilters, postActionContext.Result); 


catch (ThreadAbortException) { 
// This type of exception occurs as a result of Response.Redirect(), but we special-case so that 
// the filters don't see this as an error. 
throw; 

catch (Exception ex) { 
// something blew up, so execute the exception filters 
ExceptionContext exceptionContext = InvokeExceptionFilters(controllerContext, filterInfo.ExceptionFilters, ex); 
if (!exceptionContext.ExceptionHandled) { 
throw; 

InvokeActionResult(controllerContext, exceptionContext.Result); 


return true; 


// notify controller that no method matched 
return false; 


這個方法里面的內(nèi)容不可能一次講完的,我們看看 ControllerDescriptor controllerDescriptor = GetControllerDescriptor(controllerContext); 
很明顯ControllerDescriptor是Controller實(shí)例的一個包裝類。 

復(fù)制代碼代碼如下:


protected virtual ControllerDescriptor GetControllerDescriptor(ControllerContext controllerContext) { 
Type controllerType = controllerContext.Controller.GetType(); 
ControllerDescriptor controllerDescriptor = DescriptorCache.GetDescriptor(controllerType, () => new ReflectedControllerDescriptor(controllerType)); 
return controllerDescriptor; 


從這個方法中,我們可以知道實(shí)際返回的是一個ReflectedControllerDescriptor實(shí)例,它是ControllerDescriptor的子類,DescriptorCache.GetDescriptor(...)好像是從緩存中獲取的啊,讓我們證實(shí)一下吧,先來看看ControllerDescriptorCache的GetDescriptor方法: 

復(fù)制代碼代碼如下:


internal sealed class ControllerDescriptorCache : ReaderWriterCache<Type, ControllerDescriptor> { 
public ControllerDescriptor GetDescriptor(Type controllerType, Func<ControllerDescriptor> creator) { 
return FetchOrCreateItem(controllerType, creator); 


FetchOrCreateItem方法很簡單,從緩存中獲取ControllerDescriptor ,如果沒有就創(chuàng)建并加入緩存然后在返回,緩存實(shí)現(xiàn)方式其實(shí)就是一個字典Dictionary<TKey, TValue>。 
現(xiàn)在看看ReflectedControllerDescriptor的夠著函數(shù)是否有什么特別之處: 
_controllerType = controllerType; 
_selector = new ActionMethodSelector(_controllerType); 
怎么又有ActionMethodSelector這個東東啊,其構(gòu)造函數(shù)如下 

復(fù)制代碼代碼如下:


public ActionMethodSelector(Type controllerType) { 
ControllerType = controllerType; 
PopulateLookupTables(); 

private void PopulateLookupTables() { 
MethodInfo[] allMethods = ControllerType.GetMethods(BindingFlags.InvokeMethod | BindingFlags.Instance | BindingFlags.Public); 
MethodInfo[] actionMethods = Array.FindAll(allMethods, IsValidActionMethod); 
AliasedMethods = Array.FindAll(actionMethods, IsMethodDecoratedWithAliasingAttribute); 
NonAliasedMethods = actionMethods.Except(AliasedMethods).ToLookup(method => method.Name, StringComparer.OrdinalIgnoreCase); 


這個方法很簡單,找出ControllerType的所有實(shí)例、共有方法,然后在過濾調(diào)不是Action的,最后吧這些Action方法分成兩部分,一部分有別名,一部分沒有別名。 
現(xiàn)在我們已經(jīng)得到了ControllerDescriptor實(shí)例,下面應(yīng)該來看看ActionDescriptor actionDescriptor = FindAction(controllerContext, controllerDescriptor, actionName);這句代碼了;同樣我們可以確認(rèn)ActionDescriptor實(shí)際上一個Action的包裝類。 
protected virtual ActionDescriptor FindAction(ControllerContext controllerContext, ControllerDescriptor controllerDescriptor, string actionName)這個方法實(shí)際上就是調(diào)用 
ControllerDescriptor類FindAction方法,讓我們看看你ReflectedControllerDescriptor的FindAction方法,該方法很簡單,組要就2句代碼: 

復(fù)制代碼代碼如下:


MethodInfo matched = _selector.FindActionMethod(controllerContext, actionName); 
return new ReflectedActionDescriptor(matched, actionName, this); 


_selector.FindActionMethod(controllerContext, actionName); 這句就是找到我們需要Action對應(yīng)的MethodInfo。 

復(fù)制代碼代碼如下:


public MethodInfo FindActionMethod(ControllerContext controllerContext, string actionName) { 
List<MethodInfo> methodsMatchingName = GetMatchingAliasedMethods(controllerContext, actionName); 
methodsMatchingName.AddRange(NonAliasedMethods[actionName]); 
List<MethodInfo> finalMethods = RunSelectionFilters(controllerContext, methodsMatchingName); 

switch (finalMethods.Count) { 
case 0: 
return null; 

case 1: 
return finalMethods[0]; 

default: 
throw CreateAmbiguousMatchException(finalMethods, actionName); 


循環(huán)每個MethodInfo,查找它們的自定義的ActionMethodSelectorAttribute特性,如果有只返回驗(yàn)證通過的特性。看到ReflectedAttributeCache.GetActionMethodSelectorAttributes(methodInfo)這樣的代碼感覺由于緩存有關(guān), 

復(fù)制代碼代碼如下:


private static ReadOnlyCollection<TAttribute> GetAttributes<TMemberInfo, TAttribute>(ConcurrentDictionary<TMemberInfo, ReadOnlyCollection<TAttribute>> lookup, TMemberInfo memberInfo) 
where TAttribute : Attribute 
where TMemberInfo : MemberInfo { 
return lookup.GetOrAdd(memberInfo, mi => new ReadOnlyCollection<TAttribute>((TAttribute[])memberInfo.GetCustomAttributes(typeof(TAttribute), inherit: true))); 


ReflectedAttributeCache這個類有幾個緩存字典: 
ConcurrentDictionary<MethodInfo, ReadOnlyCollection<ActionMethodSelectorAttribute>> 
ConcurrentDictionary<MethodInfo, ReadOnlyCollection<ActionNameSelectorAttribute>> 
ConcurrentDictionary<MethodInfo, ReadOnlyCollection<FilterAttribute>> 
ConcurrentDictionary<Type, ReadOnlyCollection<FilterAttribute>> 
默認(rèn)實(shí)現(xiàn)ActionMethodSelectorAttribute類主要有以下幾個 
AcceptVerbsAttribute 
HttpDeleteAttribute 
HttpGetAttribute 
HttpPostAttribute 
HttpPutAttribute 
NonActionAttribute 
AcceptVerbsAttribute 
剩下的就是直接實(shí)例一個ReflectedActionDescriptor對象了,這個也沒什么特殊,只是里面有一個驗(yàn)證方法 

復(fù)制代碼代碼如下:


if (validateMethod) { 
string failedMessage = VerifyActionMethodIsCallable(methodInfo); 
if (failedMessage != null) { 
throw new ArgumentException(failedMessage, "methodInfo"); 


用來驗(yàn)證該方法是否可以執(zhí)行,以下幾種情況經(jīng)不會通過,(1)方法是靜態(tài)方法(2)方法的實(shí)例類型不是ControllerBase(3)是否包含泛型參數(shù)如 public ActionResult Index<T>()是非法的,但 public ActionResult Index(List<string> aa)是合法(4)參數(shù)中不能含有Ref和out。 
這篇文章說的很散,我們需要注意一點(diǎn)微軟在mvc里面緩存做的很好了,在前面?zhèn)€將獲取ControllerTyper時它是有緩存的,一次讀取當(dāng)前程序集所有的ControllerType,在這里提到了一個DescriptorCache 緩存每次調(diào)用的ControllerType->ReflectedControllerDescriptor,而ReflectedControllerDescriptor實(shí)例會一次去讀該Controller的所有Action方法;這里還有一個ReflectedAttributeCache,緩存每次調(diào)用MethodInfo的所有特性(ActionMethodSelectorAttribute、ActionNameSelectorAttribute、FilterAttribute),當(dāng)然FilterAttribute特性還可以在類上面。

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 激情影院在线观看 | 国产精品免费久久久久 | 欧美激情精品久久久久久久久久 | 国产九色视频在线观看 | 欧美日韩免费看 | 久久人人爽人人爽人人片av高清 | 黄色网址在线免费播放 | 久久综合婷婷香五月 | 国产亚洲精品综合一区91555 | 一本在线高清码电影 | 亚洲欧美国产高清 | 一区二区三区黄色 | 91久久国产露脸精品国产护士 | av电影在线免费 | 国产在线精品一区二区三区 | 亚洲精久 | 欧洲狠狠鲁 | 国产精品免费一区二区三区在线观看 | 精品成人一区二区三区 | 国产在线精品一区二区 | 竹内纱里奈和大战黑人 | 日本高清在线免费 | 久久国产精品久久久久 | 中国av中文字幕 | 久久精品亚洲一区二区 | 91在线色| 久久久精品精品 | 在线a| 欧美 videos粗暴 | 国产喷白浆10p | 91成人免费视频 | 少妇一级淫片免费放4p | 国产精品久久999 | 中文字幕在线看第二 | 中文字幕在线观看网址 | 99久久精品免费看国产小宝寻花 | 激情视频在线播放 | 久久99国产精品免费网站 | 欧美另类综合 | 666sao| 亚洲午夜免费电影 |