1. 程式人生 > >Var與Dynamic的區別

Var與Dynamic的區別

data runt foreach bsp www 詳情 net png 是我

1.var與dynamic的區別

  C#中的很多關鍵詞用法比較容易混淆,var和dynamic就是其中一組,但其實它們是有本質的區別的。var 在編譯階段已經確定類型,在初始化時候,必須提供初始化的值,而dynamic則可以不提供,它是在運行時才確定類型。

技術分享圖片
 1 static void Main(string[] args)
 2 {
 3     //var 在編譯階段已經確定類型
 4     var number = 3;
 5 
 6     //dynamic在編譯期間不進行任何的類型檢查,而是將類型檢查放到了運行期
 7     dynamic dyn = 3;
 8 
 9     dyn = "hello world";
10 
11     //runtime eror
12     //字符串沒有 "fn不存在的方法" 的方法,但在語法檢查時通過,不會提示語法錯誤
13     var s = dyn.fn不存在的方法();
14 
15     Console.ReadKey();
16 }
技術分享圖片

2.ExpandoObject類型的使用

當我們因為某種目的而需要創建一些臨時類型的變量時,我們可以會像下面這樣做:

1 var a = new { Name = "Paul Huang", Age = 24 };
2 Console.WriteLine("Name is {0},Age is {1}", a.Name, a.Age);

現在要介紹一種新的類型ExpandoObject,它是專為動態行為而設計的類型,用法如下:

技術分享圖片
1 static void Main(string[] args)
2 {
3     dynamic a = new ExpandoObject();  
4     a.Name = "Paul Huang";  
5     a.Age = 24; 
6     Console.WriteLine("Name is {0},Age is {1}", a.Name, a.Age);
7  }
技術分享圖片

實際上,ExpandoObject類顯式實現了 IDictionary<string,object> 接口,所以,我們可以知道,其實它裏面就是用一個字典來存儲動態賦值的數值的,鍵的類型為字符串,表示屬性名;值的類型為object,表示任何類型。

咱們把它裏面的字典數據輸出來:

1 IDictionary<string, object> dic = a as IDictionary<string, object>;
2 foreach (var pv in dic)
3 {
4     Console.WriteLine("Key = {0} , Value = {1}", pv.Key, pv.Value);
5 }

所以不管你如何動態設置屬性,它都可以進行解析,就是這個原因,裏面用一個字典來負責存取數據。

3.dynamic在反射時的應用

  由於dynamic在運行時才檢查類型,所以有時候會出現錯誤,因此使用它必須得法,那麽何時使用它比較方便呢?我們先定義一個Person類,然後用反射進行動態調用起Talk方法:

技術分享圖片
1 class Person
2 {
3     public void Talk(string msg)
4     {
5         Console.WriteLine(msg);
6     }
7 }
技術分享圖片

技術分享圖片
 1 class Program
 2 {
 3     static void Main(string[] args)
 4     {
 5         Type t = typeof(Person);
 6         Object obj = Activator.CreateInstance(t);
 7         t.InvokeMember("Talk", BindingFlags.InvokeMethod, null, obj, new object[] { "hello world" });
 8         Console.ReadKey();
 9     }
10 }
技術分享圖片

在反射的時候,傳統的方法調用往往比較繁瑣,而用dyanmic則非常簡化,而且直觀:

技術分享圖片
 1 class Program
 2 {
 3     static void Main(string[] args)
 4   {
 5           Type t = typeof(Person);
 6           dynamic obj = Activator.CreateInstance(t);
 7           obj.Talk("hello world");
 8      Console.ReadKey();
 9   }
10 }
技術分享圖片

轉載鏈接:https://www.cnblogs.com/JustYong/p/5113197.html

ASP.NET MVC傳遞Model到視圖的多種方式總結(二)__關於ViewBag、ViewData和TempData的實現機制與區別

在ASP.NET MVC中,視圖數據可以通過ViewBag、ViewData、TempData來訪問,其中ViewBag 是動態類型(Dynamic),ViewData 是一個字典型的(Dictionary)。

它們的定義如下:

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>
技術分享圖片

運行圖:

技術分享圖片

當然我們可以在視圖裏面這樣寫:

1 <h2>@ViewBag.Message_ViewData </h2>
2 <h2>@ViewData["Message_ViewBag"]</h2>

運行結果是一樣的,這裏表示它們倆是互通的。

ViewBag和ViewData的區別:

使用ViewBag

ViewBag 不再是字典的鍵值對結構,而是 dynamic 動態類型,它會在程序運行的時候動態解析。
控制器代碼:

技術分享圖片
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

Var與Dynamic的區別