asp.net core mvc剖析:動作執行

来源:http://www.cnblogs.com/dxp909/archive/2017/03/25/6617200.html
-Advertisement-
Play Games

緊跟上一篇文章。通過路由和動作匹配後,最終會得到跟當前請求最匹配的一個ActionDescriptor,然後通過IActionInvoker執行動作。 我們先來看一下IActionInvoker如何得到,代碼如下: 從上面的代碼可以看到,一個IActionInvoker是通過IActionInvok ...


緊跟上一篇文章。通過路由和動作匹配後,最終會得到跟當前請求最匹配的一個ActionDescriptor,然後通過IActionInvoker執行動作。

我們先來看一下IActionInvoker如何得到,代碼如下:

context.Handler = (c) =>
            {
                var routeData = c.GetRouteData();
                //根據actiondescriptor實例化ActionContext對象
                var actionContext = new ActionContext(context.HttpContext, routeData, actionDescriptor);
                if (_actionContextAccessor != null)
                {
                    _actionContextAccessor.ActionContext = actionContext;
                }
                //創建IActionInvoker
                var invoker = _actionInvokerFactory.CreateInvoker(actionContext);
                if (invoker == null)
                {
                    throw new InvalidOperationException(
                        Resources.FormatActionInvokerFactory_CouldNotCreateInvoker(
                            actionDescriptor.DisplayName));
                }
                //執行invoker處理請求
                return invoker.InvokeAsync();
            };

  從上面的代碼可以看到,一個IActionInvoker是通過IActionInvokerFactory創建,框架里該介面實現類是ActionInvokerFactory,該類CreateInvoker方法實現代碼如下:

public IActionInvoker CreateInvoker(ActionContext actionContext)
        {
            var context = new ActionInvokerProviderContext(actionContext);

            foreach (var provider in _actionInvokerProviders)
            {
                provider.OnProvidersExecuting(context);
            }

            for (var i = _actionInvokerProviders.Length - 1; i >= 0; i--)
            {
                _actionInvokerProviders[i].OnProvidersExecuted(context);
            }

            return context.Result;
        }

  在方法中,首先實例化一個ActionInvokerProviderContext,然後調用IActionInvokerProvider來設置context.Result,context.Result就是一個IActionInvoker,所以我們跟蹤下框架中IActionInvokerProvider實現類,來看看它裡面是如何工作的。框架中的提供了一個實現類ControllerActionInvokerProvider,在OnProvidersExecuting方法中創建了IActionInvoker對象,代碼如下:

public void OnProvidersExecuting(ActionInvokerProviderContext context)
        {
            if (context == null)
            {
                throw new ArgumentNullException(nameof(context));
            }

            var actionDescriptor = context.ActionContext.ActionDescriptor as ControllerActionDescriptor;

            if (actionDescriptor != null)
            {
                var controllerContext = new ControllerContext(context.ActionContext);
                // PERF: These are rarely going to be changed, so let's go copy-on-write.
                controllerContext.ValueProviderFactories = new CopyOnWriteList<IValueProviderFactory>(_valueProviderFactories);
                controllerContext.ModelState.MaxAllowedErrors = _maxModelValidationErrors;
                
                var cacheState = _controllerActionInvokerCache.GetState(controllerContext);

                context.Result = new ControllerActionInvoker(
                    _controllerFactory,
                    _argumentBinder,
                    _logger,
                    _diagnosticSource,
                    controllerContext,
                    cacheState.Filters,
                    cacheState.ActionMethodExecutor);
            }
        }

  從上面的代碼我們可以看到,最終就是實例化了一個ControllerActionInvoker對象,該類構造方法前幾個參數不再介紹了,重點是最後兩個參數,一個是IFilterMetadata[],表示跟當前動作相關聯的過濾器信息集合,一個是ObjectMethodExecutor,從名字上可以看出,這個就是控制器方法的執行器。而這兩個參數來自cacheState,這個對象是通過調用_controllerActionInvokerCache.GetState(controllerContext)得到的,它是一個ControllerActionInvokerState結構體類型,定義如下:

 public struct ControllerActionInvokerState
        {
            public ControllerActionInvokerState(
                IFilterMetadata[] filters,
                ObjectMethodExecutor actionMethodExecutor)
            {
                Filters = filters;
                ActionMethodExecutor = actionMethodExecutor;
            }
            //動作過濾器集合
            public IFilterMetadata[] Filters { get; }
            //方法執行器
            public ObjectMethodExecutor ActionMethodExecutor { get; set; }
        }

  cacheState創建的過程如下:

 public ControllerActionInvokerState GetState(ControllerContext controllerContext)
        {
            var cache = CurrentCache;
            var actionDescriptor = controllerContext.ActionDescriptor;

            IFilterMetadata[] filters;
            Entry cacheEntry;
            if (!cache.Entries.TryGetValue(actionDescriptor, out cacheEntry))
            {
                var filterFactoryResult = FilterFactory.GetAllFilters(_filterProviders, controllerContext);
                filters = filterFactoryResult.Filters;

                var executor = ObjectMethodExecutor.Create(
                    actionDescriptor.MethodInfo,
                    actionDescriptor.ControllerTypeInfo);

                cacheEntry = new Entry(filterFactoryResult.CacheableFilters, executor);
                cacheEntry = cache.Entries.GetOrAdd(actionDescriptor, cacheEntry);
            }
            else
            {
                // Filter instances from statically defined filter descriptors + from filter providers
                filters = FilterFactory.CreateUncachedFilters(_filterProviders, controllerContext, cacheEntry.FilterItems);
            }

            return new ControllerActionInvokerState(filters, cacheEntry.ActionMethodExecutor);
        }

    主要看下第10行到第15行代碼。首先通過FilterFactory獲取到跟當前動作相關聯的過濾器信息集合,然後通過ObjectMethodExecutor.Create創建一個ObjectMethodExecutor對象,創建好後進行緩存。

  其實到這裡我們已經知道IActionInvokder就是一個ControllerActionInvoker,然後調用該對象的InvokeAsync方法開始動作執行。最新版本實現採用了狀態機(如果概念上錯誤,歡迎大家拍磚指正)的特點,執行時在不同狀態之間進行切換,最終完成處理。而這部分工作就是在Next方法里實現的。方法真正執行實在State.ActionInside狀態時執行,在此狀態下通過調用InvokeActionMethodAsync完成控制器方法的執行,主要代碼如下:

                //下麵的代碼就是根據方法返回值類型不同,實現不同的邏輯
                var returnType = executor.MethodReturnType;
                if (returnType == typeof(void))
                {
                    executor.Execute(controller, orderedArguments);
                    result = new EmptyResult();
                }
                else if (returnType == typeof(Task))
                {
                    await (Task)executor.Execute(controller, orderedArguments);
                    result = new EmptyResult();
                }
                else if (executor.TaskGenericType == typeof(IActionResult))
                {
                    result = await (Task<IActionResult>)executor.Execute(controller, orderedArguments);
                    if (result == null)
                    {
                        throw new InvalidOperationException(
                            Resources.FormatActionResult_ActionReturnValueCannotBeNull(typeof(IActionResult)));
                    }
                }
                else if (executor.IsTypeAssignableFromIActionResult)
                {
                    if (_executor.IsMethodAsync)
                    {
                        result = (IActionResult)await _executor.ExecuteAsync(controller, orderedArguments);
                    }
                    else
                    {
                        result = (IActionResult)_executor.Execute(controller, orderedArguments);
                    }

                    if (result == null)
                    {
                        throw new InvalidOperationException(
                            Resources.FormatActionResult_ActionReturnValueCannotBeNull(_executor.TaskGenericType ?? returnType));
                    }
                }
                else if (!executor.IsMethodAsync)
                {
                    var resultAsObject = executor.Execute(controller, orderedArguments);
                    result = resultAsObject as IActionResult ?? new ObjectResult(resultAsObject)
                    {
                        DeclaredType = returnType,
                    };
                }
                else if (executor.TaskGenericType != null)
                {
                    var resultAsObject = await executor.ExecuteAsync(controller, orderedArguments);
                    result = resultAsObject as IActionResult ?? new ObjectResult(resultAsObject)
                    {
                        DeclaredType = executor.TaskGenericType,
                    };
                }
                else
                {
                    // This will be the case for types which have derived from Task and Task<T> or non Task types.
                    throw new InvalidOperationException(Resources.FormatActionExecutor_UnexpectedTaskInstance(
                        executor.MethodInfo.Name,
                        executor.MethodInfo.DeclaringType));
                }

                _result = result;
             
            

        }

  無論是哪種返回值類型,都是通過調用executor.Execute完成控制器方法調用,並獲取結果,executor就是上面提到的ObjectMethodExecutor,我們只看下Execute方法實現:

public object Execute(object target, object[] parameters)
        {
            return _executor(target, parameters);
        }

  這裡面又是一個_executor,它是一個delegate object ActionExecutor(object target, object[] parameters)委托類型,它是在實例化ObjectMethodInvoker時通過GetExecutor方法創建的,採用的是動態lambda表達式,有了這個委托對象,就可以執行控制器方法了。

    方法執行後,會得到方法返回結果,然後進入Result相關狀態流程,直到State.ResultInside狀態時,調用InvokeResultAsync方法執行動作結果,具體代碼如下:

protected async Task InvokeResultAsync(IActionResult result)
        {
            var actionContext = _actionContext;

            _diagnosticSource.BeforeActionResult(actionContext, result);

            try
            {
                await result.ExecuteResultAsync(actionContext);
            }
            finally
            {
                _diagnosticSource.AfterActionResult(actionContext, result);
            }
        }

  到這裡一個控制器動作就算執行完了,需要指出的是文章只介紹了主體流程,還有很多其他執行狀態文章中沒提到。


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

-Advertisement-
Play Games
更多相關文章
  • 一、什麼是主鍵、外鍵: 關係型資料庫中的一條記錄中有若幹個屬性,若其中某一個屬性組(註意是組)能唯一標識一條記錄,該屬性組就可以成為一個主鍵比如 : 學生表(學號,姓名,性別,班級) 其中每個學生的學號是唯一的,學號就是一個主鍵 用戶表(用戶名、密碼、登錄級別) 其中用戶名是唯一的, 用戶名就是一個 ...
  • 程式中的變數名、常量名、類名、方法名,都叫做標識符。C#有一套標識符的命名規則,如果命名時不遵守規則,就會出錯。這套規則簡單說有下麵三條: ①標識符只能由英文字母、數字和下劃線組成,不能包含空格和其他字元。 錯誤的標識符聲明: ②變數名不能用數字開頭。 錯誤的標識符聲明: ③不能用關鍵字當變數名。 ...
  • 跟著師父一直在做codefirst的開發,最近有個新需求,就是需要人家的資料庫,然後來開發,現在出現問題了。整理如下 目前有個現成的我們之前的codefirst的工程代碼,我記得師父說過,根據資料庫生成model的做法是: 在指定文件夾,右鍵添加-->新建項--> 這裡先選擇數據,然後是ADO.NE ...
  • 首先需要明白一個原理: 游戲畫面(動畫)是由一幀幀的圖片載入完成的,並且是連續的。也就是說,在一個時間節點上,一定存在一幀圖片。不同的是每幀的圖片存在的時間不同,如果每幀圖片存在的時間比較短,也即單位時間內幀數比較多的話(幀率大),畫面越流暢。下麵連續的表示圖片的流動,圖片的寬度表示該圖片存在的時間 ...
  • using System;using System.Collections.Generic;using System.ComponentModel;using System.Data;using System.Drawing;using System.Linq;using System.Runtim ...
  • 測試環境 操作系統:Windows8.1 開發工具:Unity5.5.2 1、新建一個測試項目,觀測引用System.Xml與Mono.Xml解析文件正確性,與打包後APK體積大小。 2、Mono.Xml 用例 編譯日誌可以觀測到沒有引用System.Xml.dll(該文件約1mb) 3、Syste ...
  • 1.VisualStudio2017設置版權 a 在團隊開發或者公司開發中,我們一般都喜歡給自己所創建的類或者介面以及其它模板設置版權說明,但是每個類一個一個的去加又是非常的費勁,所以一般情況下我們都是設置模板來實現它,當您在VS中創建類或者介面等的時候自動將這些註釋添加到您所在的類或者介面中。 b ...
  • 項目中的代碼洋洋灑灑寫了很多,最近回過頭來看看,能精簡的地方太多了。WPF MVVM是個非常實用的模式。但前提是控制項需要支持。等等,還有不支持binding的控制項麽?基礎的控制項當然不在此列,然而實踐中常常會遇到需要組合的控制項,將一組控制項放在一起完成一個基本功能。還有控制項需要根據不同的情況改變顯示形式 ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...