C#高性能動(dòng)態(tài)獲取對(duì)象屬性值的步驟
動(dòng)態(tài)獲取對(duì)象的性能值,這個(gè)在開(kāi)發(fā)過(guò)程中經(jīng)常會(huì)遇到,這里我們探討一下何如高性能的獲取屬性值。為了對(duì)比測(cè)試,我們定義一個(gè)類People
public class People { public string Name { get; set; } }
然后通過(guò)直接代碼調(diào)用方式來(lái)取1千萬(wàn)次看要花多少時(shí)間:
private static void Directly() { People people = new People { Name = "Wayne" }; Stopwatch stopwatch = Stopwatch.StartNew(); for (int i = 0; i < 10000000; i++) { object value = people.Name; } stopwatch.Stop(); Console.WriteLine("Directly: {0}ms", stopwatch.ElapsedMilliseconds); }
大概花了37ms:
反射
通過(guò)反射來(lái)獲取對(duì)象的屬性值,這應(yīng)該是大家常用的方式,但這種方式的性能比較差。接下來(lái)我們來(lái)看看同樣取1千萬(wàn)次需要多少時(shí)間:
private static void Reflection() { People people = new People { Name = "Wayne" }; Type type = typeof(People); PropertyInfo property = type.GetProperty("Name"); Stopwatch stopwatch = Stopwatch.StartNew(); for (int i = 0; i < 10000000; i++) { object value = property.GetValue(people); } stopwatch.Stop(); Console.WriteLine("Reflection: {0}ms", stopwatch.ElapsedMilliseconds); }
大概花了1533ms,果然要慢很多:
那既然反射慢,那還有沒(méi)有其它方式呢?
動(dòng)態(tài)構(gòu)建Lambda
我們知道可以動(dòng)態(tài)構(gòu)建Linq的Lambda表達(dá)式,然后通過(guò)編譯后得到一個(gè)委托,如果能動(dòng)態(tài)構(gòu)建返回屬性值的委托,就可以取到值了。所以我們想辦法構(gòu)建一個(gè)像這樣的委托:
Func<People, object> getName = m => m.Name;
接下來(lái)我們就通過(guò)Expression來(lái)構(gòu)建:
private static void Lambda() { People people = new People { Name = "Wayne" }; Type type = typeof(People); var parameter = Expression.Parameter(type, "m");//參數(shù)m PropertyInfo property = type.GetProperty("Name"); Expression expProperty = Expression.Property(parameter, property.Name);//取參數(shù)的屬性m.Name var propertyDelegateExpression = Expression.Lambda(expProperty, parameter);//變成表達(dá)式 m => m.Name var propertyDelegate = (Func<People, object>)propertyDelegateExpression.Compile();//編譯成委托 Stopwatch stopwatch = Stopwatch.StartNew(); for (int i = 0; i < 10000000; i++) { object value = propertyDelegate.Invoke(people); } stopwatch.Stop(); Console.WriteLine("Lambda:{0}ms", stopwatch.ElapsedMilliseconds); }
然后我們測(cè)試一下,大概花了138ms,性能要比反射好非常多:
委托調(diào)用
雖然動(dòng)態(tài)構(gòu)建Lambda的性能已經(jīng)很好了,但還是更好嗎?畢竟比直接調(diào)用還是差了一些,要是能直接調(diào)用屬性的取值方法就好了。
在C#中,可讀屬性都有一個(gè)對(duì)應(yīng)的get_XXX()的方法,可以通過(guò)調(diào)用這個(gè)方法來(lái)取得對(duì)應(yīng)屬性的值??梢允褂肧ystem.Delegate.CreateDelegate創(chuàng)建一個(gè)委托來(lái)調(diào)用這個(gè)方法。
- 通過(guò)委托調(diào)用方法來(lái)取得屬性值
我們定義一個(gè)MemberGetDelegate的委托,然后通過(guò)它來(lái)調(diào)用取值方法:
delegate object MemberGetDelegate(People p); private static void Delegate() { People people = new People { Name = "Wayne" }; Type type = typeof(People); PropertyInfo property = type.GetProperty("Name"); MemberGetDelegate memberGet = (MemberGetDelegate)System.Delegate.CreateDelegate(typeof(MemberGetDelegate), property.GetGetMethod()); Stopwatch stopwatch = Stopwatch.StartNew(); for (int i = 0; i < 10000000; i++) { object value = memberGet(people); } stopwatch.Stop(); Console.WriteLine("Delegate: {0}ms", stopwatch.ElapsedMilliseconds); }
然后我們測(cè)試一下,大概花了38ms,性能幾乎與直接調(diào)用一致:
最后做一個(gè)簡(jiǎn)單的封裝,緩存一下創(chuàng)建的Delegate
public class PropertyValue<T> { private static ConcurrentDictionary<string, MemberGetDelegate> _memberGetDelegate = new ConcurrentDictionary<string, MemberGetDelegate>(); delegate object MemberGetDelegate(T obj); public PropertyValue(T obj) { Target = obj; } public T Target { get; private set; } public object Get(string name) { MemberGetDelegate memberGet = _memberGetDelegate.GetOrAdd(name, BuildDelegate); return memberGet(Target); } private MemberGetDelegate BuildDelegate(string name) { Type type = typeof(T); PropertyInfo property = type.GetProperty(name); return (MemberGetDelegate)Delegate.CreateDelegate(typeof(MemberGetDelegate), property.GetGetMethod()); } }
這樣使用起來(lái)就方便多了
People people = new People { Name = "Wayne" }; PropertyValue<People> propertyValue = new PropertyValue<People>(people); object value = propertyValue.Get("Name");
以上就是C#高性能動(dòng)態(tài)獲取對(duì)象屬性值的步驟的詳細(xì)內(nèi)容,更多關(guān)于c# 獲取對(duì)象屬性值的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
C#使用SqlDataAdapter對(duì)象獲取數(shù)據(jù)的方法
這篇文章主要介紹了C#使用SqlDataAdapter對(duì)象獲取數(shù)據(jù)的方法,結(jié)合實(shí)例形式較為詳細(xì)的分析了SqlDataAdapter對(duì)象獲取數(shù)據(jù)具體步驟與相關(guān)使用技巧,需要的朋友可以參考下2016-02-02C#在復(fù)雜多線程環(huán)境下使用讀寫鎖同步寫入文件
這篇文章介紹了C#在復(fù)雜多線程環(huán)境下使用讀寫鎖同步寫入文件的方法,文中通過(guò)示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-04-04C#定制Excel界面并實(shí)現(xiàn)與數(shù)據(jù)庫(kù)交互的方法
這篇文章主要介紹了C#定制Excel界面并實(shí)現(xiàn)與數(shù)據(jù)庫(kù)交互的方法的相關(guān)資料,需要的朋友可以參考下2015-11-11C#把UNICODE編碼轉(zhuǎn)換為GB編碼的實(shí)例
下面小編就為大家?guī)?lái)一篇C#把UNICODE編碼轉(zhuǎn)換為GB編碼的實(shí)例。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-01-01C#WinForm實(shí)現(xiàn)多語(yǔ)言切換的示例
本文主要介紹了C#WinForm實(shí)現(xiàn)多語(yǔ)言切換的示例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-01-01