dynamic關鍵字才出來的時候,覺得真是沒什么用,誰總是和com交互來交互去啊,唯恐避之不及啊。
后來逐漸算是有了一些使用心得,發現這貨還真是犀利啊,故在此舉幾個例子,起拋磚引玉之用。
1.替代XXX.GetType().GetProperty("YYY").GetValue(XXX)
static object GetPerson() { return new Person { Name = "Leo" }; }
有時候難免會遇到這種返回object的倒霉代碼(特別是跟反射有關的時候),這時我們又要訪問其中的某個屬性,那個費勁啊,現在有了dynamic感覺好多了。
object objPerson = GetPerson(); var objName = objPerson.GetType().GetProperty("Name").GetValue(objPerson); Console.WriteLine(objName); dynamic dynPerson = GetPerson(); var dynName = dynPerson.Name; Console.WriteLine(dynName);
另一個好處是性能會得到一程度的提升:
Watch = new Stopwatch(); Watch.Start(); for (int i = 0; i < 1000000; i++) { objName = objPerson.GetType().GetProperty("Name").GetValue(objPerson); } Watch.Stop(); Console.WriteLine(Watch.Elapsed); Watch.Restart(); for (int i = 0; i < 1000000; i++) { dynName = dynPerson.Name; } Watch.Stop(); Console.WriteLine(Watch.Elapsed);
大致結果如下圖,還是快了很多的:
2.拯救接手接口沒設計好的代碼的倒霉孩子
比如這里有N個WCF服務,返回了N個對象的集合,這幾個對象沒啥關系,其實又有一點關系,倒霉孩子又不會讓Entity Framework生成的類自動繼承某個接口(本文里用本地方法代替WCF服務)。
這里來舉一個例子,首先有下面2個倒霉的類,同樣string類型的name是可以提取接口的(這里真的合適提取么……),同樣名稱但不同類型的ID,完全無關的Age和Price。
public class Person { public int ID { get; set; } public string Name { get; set; } public int Age { get; set; } public static List<Person> GetPersonList() { return new List<Person> { new Person{ Name = "Leo1" , Age = 10 }, new Person{ Name = "Leo2" , Age = 20 }, new Person{ Name = "Leo3" , Age= 30 } }; } } public class Car { public Guid ID { get; set; } public string Name { get; set; } public double Price { get; set; } public static List<Car> GetCarList() { return new List<Car> { new Car{ Name = "Focus1" , Price = 100 }, new Car{ Name = "Focus2" , Price = 200 }, new Car{ Name = "Focus3" , Price = 300 } }; } }
我用2個static方法返回不同類型的List<T>來模擬WCF中最普通的調用。
static void Main(string[] args) { List<dynamic> list = new List<dynamic>(); //用本地方法替代WCF服務,您假裝是通過WCF獲取的list Person.GetPersonList().ForEach((p) => list.Add(p)); TestDynamic2(list,"Leo2"); list = new List<dynamic>(); //用本地方法替代WCF服務,您假裝是通過WCF獲取的list Car.GetCarList().ForEach((c) => list.Add(c)); TestDynamic2(list,"Focus3"); Console.ReadKey(); } private static void TestDynamic2(List<dynamic> list,string name) { //可以無差別的使用ID和Name屬性 dynamic first = list.OrderBy(d => d.ID).FirstOrDefault(d => d.Name.Contains(name)); //差別對待不同的屬性,這里供參考,不建議這么寫,這會導致依賴具體的類型 if (first is Person) { Console.WriteLine(first.Age); } else { Console.WriteLine(first.Price); } }
本文提供了使用dynamic的一點心得,如果有不對的對方,還望各位不吝指出,非常感謝!