在ASP.NET MVC中,視圖數據能夠經過ViewBag、ViewData、TempData來訪問,其中ViewBag 是動態類型(Dynamic),ViewData 是一個字典型的(Dictionary)。html
它們的定義以下:web
1 public dynamic ViewBag { get; } 2 public ViewDataDictionary ViewData { get; set; }
控制器中代碼:數組
1 public ActionResult Index() 2 { 3 ViewBag.Message_ViewBag = "I am viewbag"; 4 ViewData["Message_ViewData"] = "I am viewdata"; 5 return View(); 6 }
視圖代碼:併發
1 @{ 2 ViewBag.Title = "主頁"; 3 } 4 5 <h2>@ViewBag.Message_ViewBag</h2> 6 <h2>@ViewData["Message_ViewData"]</h2>
運行圖:ide
固然咱們能夠在視圖裏面這樣寫:this
1 <h2>@ViewBag.Message_ViewData </h2> 2 <h2>@ViewData["Message_ViewBag"]</h2>
運行結果是同樣的,這裏表示它們倆是互通的。編碼
ViewBag和ViewData的區別:spa
使用ViewBag調試
ViewBag 再也不是字典的鍵值對結構,而是 dynamic 動態類型,它會在程序運行的時候動態解析。
控制器代碼:code
1 public ActionResult Index() 2 { 3 string[] items = new string[] {"one","two","three" }; 4 ViewBag.Items = items;// viewbag是一個新的dynamic關鍵字的封裝器 //ViewData["Items"] = items; 5 return View(); 6 }
視圖代碼:
1 <ul> 2 @foreach (dynamic p in ViewBag.Items) 3 { 4 <li>The item is: @p</li> 5 } 6 </ul>
其中dynamic p能夠用var p或者string p取代
執行效果:
使用ViewData
若是使用ViewData,則會出現以下錯誤:
這時若是咱們但願使用ViewData,就須要咱們本身手動去將它強制轉換爲數組。經過調試,咱們能夠看到
1 string[] items = new string[] { "one", "two", "three" }; 2 ViewBag.Items = items; 3 ViewData["Items"] = items;
兩者對比
賦值後的ViewBag和ViewData都是字符串數組形式。以下圖:
只是ViewData爲object型,而ViewBag爲dynamic型。而dynamic型與object型的區別則是在使用時它會自動根據數據類型轉換,而object型則須要咱們本身去強制轉換。好比上面咱們遍歷ViewBag.Items時,它自動根據數據類型轉換,而ViewData則須要咱們強制轉換,以下:
1 @foreach (string a in (string[])ViewData["Items"]) 2 { 3 <li>The item is: @a</li> 4 }
此外,經過轉到定義咱們能夠看到:
1 [Dynamic] 2 public dynamic ViewBag { get; } 3 public ViewDataDictionary ViewData { get; set; }
這裏ViewBag只有get方法,沒有set方法,可是咱們在上面卻給ViewBag賦值了。經過反編譯發現ViewBag代碼以下:
1 [Dynamic] 2 public object ViewBag 3 { 4 [return: Dynamic] 5 get 6 { 7 Func<ViewDataDictionary> viewDataThunk = null; 8 if (this._dynamicViewDataDictionary == null) 9 { 10 if (viewDataThunk == null) 11 { 12 viewDataThunk = () => this.ViewData; 13 } 14 this._dynamicViewDataDictionary = new DynamicViewDataDictionary(viewDataThunk); 15 } 16 return this._dynamicViewDataDictionary; 17 } 18 }
不難看出ViewBag返回的是_dynamicViewDataDictionary,繼續跟蹤發現_dynamicViewDataDictionary屬於 DynamicViewDataDictionary類,其代碼以下:
1 internal sealed class DynamicViewDataDictionary : DynamicObject 2 { 3 // Fields 4 private readonly Func<ViewDataDictionary> _viewDataThunk; 5 6 // Methods 7 public DynamicViewDataDictionary(Func<ViewDataDictionary> viewDataThunk); 8 public override IEnumerable<string> GetDynamicMemberNames(); 9 public override bool TryGetMember(GetMemberBinder binder, out object result); 10 public override bool TrySetMember(SetMemberBinder binder, object value); 11 12 // Properties 13 private ViewDataDictionary ViewData { get; } 14 15 其中有TryGetMember和TrySetMember方法,點開這兩個方法: 16 public override bool TrySetMember(SetMemberBinder binder, object value) 17 { 18 this.ViewData[binder.Name] = value; 19 return true; 20 } 21 22 public override bool TryGetMember(GetMemberBinder binder, out object result) 23 { 24 result = this.ViewData[binder.Name]; 25 return true; 26 } 27 }
發現ViewBag其實本質就是ViewData,只是多了層Dynamic控制。因此,使用何種方式徹底取決於你我的的愛好。
TempData的使用
同ViewData和ViewBag同樣,TempData也能夠用來向視圖傳遞數據。只是ViewData和ViewBag的生命週期和View相同,只對當前View有用。而TempData則能夠在不一樣的Action中進行傳值,相似webform裏的Seesion。以下:
1 public ActionResult Index() 2 { 3 ViewBag.hello = "hello,this is viewBag"; 4 ViewData["hi"] = "hi,this is viewData"; 5 TempData["abc"] = "this is tempdata"; 6 return View(); 7 }
而後在About視圖裏面調用:
1 <h2>關於</h2> 2 <p> 3 @ViewBag.hello 4 @ViewData["key"] 5 @TempData["abc"] 6 </p>
頁面效果以下:
這裏只獲取到了TempData["abc"]的值,可是TempData的值在取了一次後則會自動刪除,這時我再刷新頁面,則TempData["abc"]爲Null了。
經過反編譯查看代碼,發現TempData數據在調用後則會自動刪除。詳情請參考: http://www.cnblogs.com/tristanguo/archive/2009/04/06/1430062.html
(TempData默認是使用Session來存儲臨時數據的,TempData中存放的數據只一次訪問中有效,一次訪問完後就會刪除了的。這個一次訪問指的是一個請求到下一個請求,由於在下一個請求到來以後,會從Session中取出保存在裏面的TempData數據並賦值給TempData,而後將數據從Session中刪除。咱們看一下ASP.NET MVC Preview5源碼:
也就是說TempData只保存到下一個請求中,下一個請求完了以後,TempData就會被刪除了。注意這裏TempData使用Session來作存儲的,Session是對應特定用戶的,因此並不存在併發問題。
若想TempData中的數據在訪問下一個請求後不被刪除,則可使用TempData.Keep()方法。)
其它視圖注意事項
<li>The item is: @Html.Raw(p)</li>表示對p不進行HTML編碼。
控制器能夠返回本視圖,也能夠返回其餘視圖,以下所示:
1 public ActionResult Index() 2 { 3 ViewBag.Message_ViewBag = "I am viewbag"; 4 ViewData["Message_ViewData"] = "I am viewdata"; 5 return View("About"); 6 }
當咱們須要返回指定徹底不一樣目錄結構中的視圖時,能夠這樣使用~符號來提供視圖的完整路徑來返回:
1 return View("~/Views/Home/About.cshtml");
參考連接:https://www.cnblogs.com/bianlan/archive/2013/01/11/2857105.html