.NET Core開發日誌——視圖與頁面

来源:https://www.cnblogs.com/kenwoo/archive/2018/08/26/9533725.html
-Advertisement-
Play Games

當一個Action完成它的任務後,通常需要返回一個實現IActionResult的對象,而最常見的就是View或者ViewResult,所謂的視圖對象。那麼視圖與最終所看到的頁面之間的聯繫又是怎樣形成的,這便是本文想要探討的問題。 在ResourceInvoker類之中,可以找到下列的代碼。這些代碼 ...


當一個Action完成它的任務後,通常需要返回一個實現IActionResult的對象,而最常見的就是View或者ViewResult,所謂的視圖對象。那麼視圖與最終所看到的頁面之間的聯繫又是怎樣形成的,這便是本文想要探討的問題。

在ResourceInvoker類之中,可以找到下列的代碼。這些代碼是對返回結果——IActionResult的進一步處理。

case State.ResultInside:
    {
        ...

        var task = InvokeResultAsync(_result);
        if (task.Status != TaskStatus.RanToCompletion)
        {
            next = State.ResultEnd;
            return task;
        }

        goto case State.ResultEnd;
    }

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

    _diagnosticSource.BeforeActionResult(actionContext, result);
    _logger.BeforeExecutingActionResult(result);

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

IActionResult介面的實現類ViewResult中會調用ViewResultExecutor類的方法。

public override async Task ExecuteResultAsync(ActionContext context)
{
    ...

    var executor = context.HttpContext.RequestServices.GetRequiredService<IActionResultExecutor<ViewResult>>();
    await executor.ExecuteAsync(context, this);
}

ViewResultExecutor類里則需要先通過RazorViewEngine類找到對應的視圖。

public async Task ExecuteAsync(ActionContext context, ViewResult result)
{
    ...

    var viewEngineResult = FindView(context, result);
    viewEngineResult.EnsureSuccessful(originalLocations: null);

    var view = viewEngineResult.View;
    using (view as IDisposable)
    {

        await ExecuteAsync(
            context,
            view,
            result.ViewData,
            result.TempData,
            result.ContentType,
            result.StatusCode);
    }

    ...
}

RazorViewEngine類返回的結果是RazorView對象。註意其內部已包含了IRazorPage對象。

public ViewEngineResult GetView(string executingFilePath, string viewPath, bool isMainPage)
{
    ...

    var cacheResult = LocatePageFromPath(executingFilePath, viewPath, isMainPage);
    return CreateViewEngineResult(cacheResult, viewPath);
}

public ViewEngineResult FindView(ActionContext context, string viewName, bool isMainPage)
{
    ...

    var cacheResult = LocatePageFromViewLocations(context, viewName, isMainPage);
    return CreateViewEngineResult(cacheResult, viewName);
}

private ViewEngineResult CreateViewEngineResult(ViewLocationCacheResult result, string viewName)
{
    ...

    var page = result.ViewEntry.PageFactory();

    var viewStarts = new IRazorPage[result.ViewStartEntries.Count];
    for (var i = 0; i < viewStarts.Length; i++)
    {
        var viewStartItem = result.ViewStartEntries[i];
        viewStarts[i] = viewStartItem.PageFactory();
    }

    var view = new RazorView(this, _pageActivator, viewStarts, page, _htmlEncoder, _diagnosticSource);
    return ViewEngineResult.Found(viewName, view);
}

找到視圖後,ViewResultExecutor再調用其父類ViewExecutor的ExecuteAsync方法。其內部將調用RazorView類的RenderAsync方法。

protected async Task ExecuteAsync(
    ViewContext viewContext,
    string contentType,
    int? statusCode)
{
    ...

    var response = viewContext.HttpContext.Response;

    ResponseContentTypeHelper.ResolveContentTypeAndEncoding(
        contentType,
        response.ContentType,
        DefaultContentType,
        out var resolvedContentType,
        out var resolvedContentTypeEncoding);

    response.ContentType = resolvedContentType;

    if (statusCode != null)
    {
        response.StatusCode = statusCode.Value;
    }

    using (var writer = WriterFactory.CreateWriter(response.Body, resolvedContentTypeEncoding))
    {
        var view = viewContext.View;

        var oldWriter = viewContext.Writer;
        try
        {
            viewContext.Writer = writer;

            DiagnosticSource.BeforeView(view, viewContext);

            await view.RenderAsync(viewContext);

            DiagnosticSource.AfterView(view, viewContext);
        }
        finally
        {
            viewContext.Writer = oldWriter;
        }

        // Perf: Invoke FlushAsync to ensure any buffered content is asynchronously written to the underlying
        // response asynchronously. In the absence of this line, the buffer gets synchronously written to the
        // response as part of the Dispose which has a perf impact.
        await writer.FlushAsync();
    }
}

RazorView類中可以看到其核心的處理與IRazorPage的ExecuteAsync方法緊密相關。

public virtual async Task RenderAsync(ViewContext context)
{
    ...

    _bufferScope = context.HttpContext.RequestServices.GetRequiredService<IViewBufferScope>();
    var bodyWriter = await RenderPageAsync(RazorPage, context, invokeViewStarts: true);
    await RenderLayoutAsync(context, bodyWriter);
}

private async Task<ViewBufferTextWriter> RenderPageAsync(
    IRazorPage page,
    ViewContext context,
    bool invokeViewStarts)
{
    var writer = context.Writer as ViewBufferTextWriter;
    ...

    // The writer for the body is passed through the ViewContext, allowing things like HtmlHelpers
    // and ViewComponents to reference it.
    var oldWriter = context.Writer;
    var oldFilePath = context.ExecutingFilePath;

    context.Writer = writer;
    context.ExecutingFilePath = page.Path;

    try
    {
        if (invokeViewStarts)
        {
            // Execute view starts using the same context + writer as the page to render.
            await RenderViewStartsAsync(context);
        }

        await RenderPageCoreAsync(page, context);
        return writer;
    }
    finally
    {
        context.Writer = oldWriter;
        context.ExecutingFilePath = oldFilePath;
    }
}

private async Task RenderPageCoreAsync(IRazorPage page, ViewContext context)
{
    page.ViewContext = context;
    _pageActivator.Activate(page, context);

    _diagnosticSource.BeforeViewPage(page, context);

    try
    {
        await page.ExecuteAsync();
    }
    finally
    {
        _diagnosticSource.AfterViewPage(page, context);
    }
}

但當查找IRazorPage介面的實現。從RazorPageBaseRazorPage,再到RazorPage<TModel>,這些都只是抽象類,且都沒有對ExecuteAsync方法有具體實現。

源碼里找不到進一步的實現類,線索到這裡斷開了。

這時可以建立一個MVC的應用程式,編譯後找到它的bin目錄,會看到其中包含一個*.View.dll文件。

使用反編譯軟體,比如dotPeek,查看裡面的內容,會找到一些由cshtml文件生成的類。

以其中Views_Home_Index為例,其實際上為RazorPage<TModel>的一個實現類。

它內部的ExecuteAsync方法正是生成頁面內容的關鍵。

因為是VS模板自動生成的頁面,上面的代碼十分冗雜。為了更清晰地檢查核心的代碼,不妨減少下頁面的複雜度。

把index.cshtml文件內容改成如下:

@{
    ViewData["Title"] = "Home Page";
    Layout = null;
}

<p>Hello World!</p>

再次編譯後,可以看到ExecuteAsync方法的內容變成了下麵的樣子:

public virtual async Task ExecuteAsync()
{
  ((ViewDataDictionary) this.get_ViewData()).set_Item("Title", (object) "Home Page");
  ((RazorPageBase) this).set_Layout((string) null);
  ((RazorPageBase) this).BeginContext(65, 21, true);
  ((RazorPageBase) this).WriteLiteral("\r\n<p>Hello World!</p>");
  ((RazorPageBase) this).EndContext();
}

不難看出,最終展現的頁面內容便是通過RazorPageBase類的WriteLiteral方法生成的。


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

-Advertisement-
Play Games
更多相關文章
  • 1、繼承 2、抽象類 3、綜合案例---員工類系列定義 ...
  • Python中的For迴圈 1.For迴圈語句 1.Python for迴圈可以遍歷任何序列類型,如一個列表或者一個字元串。如下代碼應該能理解,依次輸出序列元素。 2.for else語句(就是在迴圈結束後,執行else的內容) 3.for迴圈中的break語句與countinue語句(條件達成時, ...
  • 一、三元表達式的使用 name = 'alex' age = 20 if name == 'alex' else 22 print(age) 二、列表推導式(聲明式編程) l = ['alex%s' %i for i in range(10) if i > 5] print(l) 三、生成器表達式 ...
  • 前言 Python 是一種極具可讀性和通用性的編程語言。Python 這個名字的靈感來自於英國喜劇團體 Monty Python,它的開發團隊有一個重要的基礎目標,就是使語言使用起來很有趣。Python 易於設置,並且是用相對直接的風格來編寫,對錯誤會提供即時反饋,對初學者而言是個很好的選擇。 Py ...
  • 我們在coding的時候,會經常遇到要求我們處理InterruptedException的情況,本文將解釋如何正確處理此異常以及背後的原因。 ...
  • 相信很多微信用戶在使用微信給朋友,同事發送相冊中的文件時,微信會顯示你手機中的視頻文件,這樣很不方便. 如果要完全不顯示視頻文件: 隨便在手機中建立一個文件夾,名字叫 ".nomedia",把視頻文件丟進去就行了. 如何隱藏你希望隱藏的視頻文件: 隨便在手機中建立一個文件夾,名字叫 ".nomedi ...
  • Python序列內置類型之元組類型詳解 1.元祖的概念 Python中的元組與列表類似,都是一個序列,不同的是元組的元素不能修改而已。 2.元組的創建 元組使用小括弧,列表使用方括弧。 註意:元組中只包含一個元素時,需要在元素後面添加逗號,否則括弧會被當作運算符使用。 3.Python元組操作 1. ...
  • 跳槽不算頻繁,但參加過不少面試(電話面試、face to face面試),面過大/小公司、互聯網/傳統軟體公司,麵糊過(眼高手低,缺乏實戰經驗,掛掉),也面過人,所幸未因失敗而氣餒,在此過程中不斷查缺補漏,養成了踏實、追本溯源、持續改進的習慣,特此將自己經歷過、構思過的一些面試題記錄下來,如果答案有 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...