ASP.NET MVC传递Model到视图的多种方式总结(二)__关于ViewBag、ViewData和TempData的实现机制与区别

     在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

相关文章
相关标签/搜索