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
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...