TempData知多少

来源:http://www.cnblogs.com/sheng-jie/archive/2017/01/20/6321800.html
-Advertisement-
Play Games

網上對TempData的總結為: 保存在session中,Controller每次執行請求時,會從session中一次獲取所有tempdata數據,保存在單獨的內部數據字典中,而後從session中清空tempdata。然後通過key從字典中獲取指定的Tempdata,每訪問一次後對應的Key就會從 ...


網上對TempData的總結為: 保存在session中,Controller每次執行請求時,會從session中一次獲取所有tempdata數據,保存在單獨的內部數據字典中,而後從session中清空tempdata。然後通過key從字典中獲取指定的Tempdata,每訪問一次後對應的Key就會從字典中刪除,因此Tempdata數據最多只能經過一次controller傳遞,並且每個元素最多只能訪問一次。

也許你不需要知道背後的原理,那下麵這張圖就能滿你所需。

若好奇它背後是怎樣實現,跟我查下源碼一探究竟:

一、Controller是何時取TempData數據

看下Controller類的ExecuteCore方法

    /// <summary>執行請求</summary>
    protected override void ExecuteCore()
    {
      this.PossiblyLoadTempData();
      try
      {
        string actionName = Controller.GetActionName(this.RouteData);
        if (this.ActionInvoker.InvokeAction(this.ControllerContext, actionName))
          return;
        this.HandleUnknownAction(actionName);
      }
      finally
      {
        this.PossiblySaveTempData();
      }
    }

    internal void PossiblyLoadTempData()
    {
      if (this.ControllerContext.IsChildAction)
        return;
      this.TempData.Load(this.ControllerContext, this.TempDataProvider);
    }

    internal void PossiblySaveTempData()
    {
      if (this.ControllerContext.IsChildAction)
        return;
      this.TempData.Save(this.ControllerContext, this.TempDataProvider);
    }

從中可以看到在請求開始時就去取TempData,在Action調用結束後去保存TempData。
為什麼要再去保存一遍呢?

二、TempDataProvider 臨時數據存儲方案

1、Controller類中,定義了TempDataProvider屬性

    /// <summary>獲取用於為下一個請求存儲數據的臨時數據提供程式對象。  </summary>
    /// <returns>臨時數據提供程式。</returns>
    public ITempDataProvider TempDataProvider
    {
      get
      {
        if (this._tempDataProvider == null)
          this._tempDataProvider = this.CreateTempDataProvider();
        return this._tempDataProvider;
      }
      set
      {
        this._tempDataProvider = value;
      }
    }

    /// <summary>創建臨時數據提供程式。</summary>
    /// <returns>臨時數據提供程式。</returns>
    protected virtual ITempDataProvider CreateTempDataProvider()
    {
      ITempDataProviderFactory service = this.Resolver.GetService<ITempDataProviderFactory>();
      if (service != null)
        return service.CreateInstance();
      return this.Resolver.GetService<ITempDataProvider>() ?? (ITempDataProvider) new SessionStatesTempDataProvider();
    }

從代碼中可知MVC中預設使用的是SessionStatesTempDataProvider來存儲臨時數據。

2、看一看SessionStatesTempDataProvider的實現

public class SessionStateTempDataProvider : ITempDataProvider
  {
    internal const string TempDataSessionStateKey = "__ControllerTempData";

    /// <summary>使用指定的控制器上下文來載入臨時數據。</summary>
    /// <returns>臨時數據。</returns>
    /// <param name="controllerContext">控制器上下文。</param>
    /// <exception cref="T:System.InvalidOperationException">檢索會話上下文時出錯。</exception>
    public virtual IDictionary<string, object> LoadTempData(ControllerContext controllerContext)
    {
      HttpSessionStateBase session = controllerContext.HttpContext.Session;
      if (session != null)
      {
        Dictionary<string, object> dictionary = session["__ControllerTempData"] as Dictionary<string, object>;
        if (dictionary != null)
        {
          session.Remove("__ControllerTempData");
          return (IDictionary<string, object>) dictionary;
        }
      }
      return (IDictionary<string, object>) new Dictionary<string, object>((IEqualityComparer<string>) StringComparer.OrdinalIgnoreCase);
    }

    /// <summary>使用指定的控制器上下文將指定的值保存在臨時數據字典中。</summary>
    /// <param name="controllerContext">控制器上下文。</param>
    /// <param name="values">值。</param>
    /// <exception cref="T:System.InvalidOperationException">檢索會話上下文時出錯。</exception>
    public virtual void SaveTempData(ControllerContext controllerContext, IDictionary<string, object> values)
    {
      if (controllerContext == null)
        throw new ArgumentNullException("controllerContext");
      HttpSessionStateBase session = controllerContext.HttpContext.Session;
      bool flag = values != null && values.Count > 0;
      if (session == null)
      {
        if (flag)
          throw new InvalidOperationException(MvcResources.SessionStateTempDataProvider_SessionStateDisabled);
      }
      else if (flag)
      {
        session["__ControllerTempData"] = (object) values;
      }
      else
      {
        if (session["__ControllerTempData"] == null)
          return;
        session.Remove("__ControllerTempData");
      }
    }
  }

從圖中可知,SessionStatesTempDataProvider暴露了LoadTempDataSaveTempData兩個方法。
其中從SaveTempDatasession["__ControllerTempData"] = (object) values;可以看出,TempData是存儲在Session中的。
其中LoadTempData方法中session.Remove("__ControllerTempData");就說明瞭從session中獲取tempdata後,對應的tempdata就從session中清空了

原來每次取完TempData後都會從Session中清空,如果TempData未曾使用,那當然要重新保存到Session中啊。(回答了上個問題)

三、TempData 何許類也?

TempData是ControllerBase中定義的屬性,TempData的類型為TempDataDictionary。
那就來看看這個類中定義的幾個核心方法。

1、 定義了索引器

    /// <summary>獲取或設置具有指定鍵的對象。</summary>
    /// <returns>具有指定鍵的對象。</returns>
    public object this[string key]
    {
      get
      {
        object obj;
        if (!this.TryGetValue(key, out obj))
          return (object) null;
        this._initialKeys.Remove(key);
        return obj;
      }
      set
      {
        this._data[key] = value;
        this._initialKeys.Add(key);
      }
    }

要註意這段代碼this._initialKeys.Remove(key);

2、再來看看取出TempData的Load方法

    /// <summary>使用指定的數據提供程式載入指定的控制器上下文。</summary>
    /// <param name="controllerContext">控制器上下文。</param>
    /// <param name="tempDataProvider">臨時數據提供程式。</param>
    public void Load(ControllerContext controllerContext, ITempDataProvider tempDataProvider)
    {
      IDictionary<string, object> dictionary = tempDataProvider.LoadTempData(controllerContext);
      this._data = dictionary != null ? new Dictionary<string, object>(dictionary, (IEqualityComparer<string>) StringComparer.OrdinalIgnoreCase) : new Dictionary<string, object>((IEqualityComparer<string>) StringComparer.OrdinalIgnoreCase);
      this._initialKeys = new HashSet<string>((IEnumerable<string>) this._data.Keys, (IEqualityComparer<string>) StringComparer.OrdinalIgnoreCase);
      this._retainedKeys.Clear();
    }

這個方法是從TempDataProvider中載入出的數據,將全部TempData保存在字典_data全局變數中,TempData的key全部保存在_initialKeys全局變數中。

3、TempData如何使用多次?

    /// <summary>將字典中的所有鍵都標記為需保留。</summary>
    public void Keep()
    {
      this._retainedKeys.Clear();
      this._retainedKeys.UnionWith((IEnumerable<string>) this._data.Keys);
    }

    /// <summary>將字典中的指定鍵標記為需保留。</summary>
    /// <param name="key">字典中要保留的鍵。</param>
    public void Keep(string key)
    {
      this._retainedKeys.Add(key);
    }

    /// <summary>返回包含與指定鍵關聯的元素的對象,不將該鍵標記為需刪除。</summary>
    /// <returns>包含與指定鍵關聯的元素的對象。</returns>
    /// <param name="key">要返回的元素的鍵。</param>
    public object Peek(string key)
    {
      object obj;
      this._data.TryGetValue(key, out obj);
      return obj;
    }

從TempData中通過索引器取值後,可以通過Keep或Peek方法,將該臨時數據保留不刪除。

4、再來看看將數據保存到TempData的Save方法

    /// <summary>使用指定的數據提供程式保存指定的控制器上下文。</summary>
    /// <param name="controllerContext">控制器上下文。</param>
    /// <param name="tempDataProvider">臨時數據提供程式。</param>
    public void Save(ControllerContext controllerContext, ITempDataProvider tempDataProvider)
    {
      this._data.RemoveFromDictionary<string, object, TempDataDictionary>((Func<KeyValuePair<string, object>, TempDataDictionary, bool>) ((entry, tempData) =>
      {
        string key = entry.Key;
        if (!tempData._initialKeys.Contains(key))
          return !tempData._retainedKeys.Contains(key);
        return false;
      }), this);
      tempDataProvider.SaveTempData(controllerContext, (IDictionary<string, object>) this._data);
    }

_data 是放Keys + Values
_initialKeys 是放Keys,取值後移除Key
_retainedKeys 是需要保留的Key
當我們根據key從索引器中讀取臨時數據時,該key從_initialKeys中移出。
Save方法首先遍歷_data:
若_initialKeys不存在該key,說明已經取值使用。
若_retainedKeys中也不存在該key,說明取值使用後並未keep。

以上兩個條件都成立時就從_data中移出該TempData。
未成立就說明臨時數據沒有使用,需重新保存到Session中。


總結

  1. Controller每次執行請求時,會從session中一次獲取所有tempdata數據,保存在單獨的內部數據字典中,而後從session中清空tempdata。
  2. 在需要的action中通過key從字典中獲取指定的Tempdata,每訪問一次後對應的Key就會從字典中刪除。因此Tempdata數據最多只能經過一次controller傳遞,並且每個元素最多只能訪問一次。
  3. Action執行完畢後數據字典中未使用的tempdata會重新保存到Session中,供下一個請求訪問。
  4. 如果tempdata使用後還想供下一個請求使用,可以通過調用TempData.Keep()TempData.Keep("key")保留至下一次請求。
  5. 還可以通過TempData.Peek()讀取,這種方式同樣會保留至下一次請求。

啰嗦了半天,TempData你懂了嗎?反正我是懂了。
還沒懂,那就回頭看看吧。



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

-Advertisement-
Play Games
更多相關文章
  • 大家都知道,現在和以前比起來,互聯網行業、軟體行業已經天差地別了。現在處處都在搞信息化建設,人人都知道互聯網思維。這樣的信息化時代,對於軟體開發者、對於軟體開發公司來說,是一個巨大的機遇。 在門外漢看來,軟體開發是機遇大、成本低,只要叫幾個程式員,就能搞出個軟體公司來。但是,事實情況是這個樣子嗎?本 ...
  • 在webuploader上傳大文件時必須配置一下,不然請求後臺處理程式時,會請求超時。出現404! <system.web> <httpRuntime maxRequestLength="2040000" useFullyQualifiedRedirectUrl="true" executionTi ...
  • 在 Windows Forms 和 WPF 應用中使用 FontAwesome 圖標 ...
  • 在C#中 “\”是特殊字元,要表示它的話需要使用“\\”。由於這種寫法不方便,C#語言提供了@對其簡化。只要在字元串前加上@即可直接使用“\”。所以上面的路徑在C#中應該表示為“Book”,@“\Tmp\Book”,@“C:\Tmp\Book”。 相對路徑使用“/”字元作為目錄的分隔字元,而絕對路徑 ...
  • 前言 上一篇講述了執行sql和配置的一些功能,這篇說明IQueryable(linq)或執行sql的查詢緩存與清理,包括擴展到將緩存存儲到Redis中。 擴展類庫源碼: github:https://github.com/skigs/EFCoreExtend 引用類庫: nuget:https:// ...
  • 這幾天老感覺不對, 總覺得少點什麼, 今天才發現, 前面 3 裡面, 在獲取Action參數信息的時候, 少解析了. 裡面還有一個比較重要的東西. 今天看也是一樣的. 在 InvokeAction() 方法裡面, 有一句代碼: 這個是用來獲取參數的. 那麼參數是不是隨便獲取呢? 在Mvc 裡面, 頁 ...
  • 16年我們公司一共開發了好幾個企業網站。最初項目經理讓我用模板引擎,我參照著網上找的模板引擎的代碼及功能,自己寫了一個。當開發了幾個企業網站之後,發現開發效率太低了,同事們用模板引擎開發的過程中,寫模板改模板要花費不少時間,有時候特殊的需求,需要加個資料庫表,或者資料庫已有的欄目或內容表加個欄位什麼 ...
  • 因為一些配置屬性比較多,存在多組屬性,因此結合xml解析、緩存技術,實現配置文化的自動解析、存入緩存、緩存依賴實時更新配置內容。 配置文件反序列化存入緩存的核心方法: public Class.Settings GetSettings() { if (HttpRuntime.Cache["setti ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...