溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

利用C#獲取對象屬性值的方法

發(fā)布時間:2020-12-09 15:22:17 來源:億速云 閱讀:2837 作者:Leah 欄目:開發(fā)技術(shù)

利用C#獲取對象屬性值的方法?很多新手對此不是很清楚,為了幫助大家解決這個難題,下面小編將為大家詳細講解,有這方面需求的人可以來學(xué)習(xí)下,希望你能有所收獲。

動態(tài)獲取對象的性能值,這個在開發(fā)過程中經(jīng)常會遇到,這里我們探討一下何如高性能的獲取屬性值。為了對比測試,我們定義一個類People

public class People
{
  public string Name { get; set; }
}

然后通過直接代碼調(diào)用方式來取1千萬次看要花多少時間:

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:

利用C#獲取對象屬性值的方法

反射
通過反射來獲取對象的屬性值,這應(yīng)該是大家常用的方式,但這種方式的性能比較差。接下來我們來看看同樣取1千萬次需要多少時間:

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,果然要慢很多:

利用C#獲取對象屬性值的方法

那既然反射慢,那還有沒有其它方式呢?

動態(tài)構(gòu)建Lambda
我們知道可以動態(tài)構(gòu)建Linq的Lambda表達式,然后通過編譯后得到一個委托,如果能動態(tài)構(gòu)建返回屬性值的委托,就可以取到值了。所以我們想辦法構(gòu)建一個像這樣的委托:

Func<People, object> getName = m => m.Name;

接下來我們就通過Expression來構(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);//變成表達式 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);
}

然后我們測試一下,大概花了138ms,性能要比反射好非常多:

利用C#獲取對象屬性值的方法

委托調(diào)用

雖然動態(tài)構(gòu)建Lambda的性能已經(jīng)很好了,但還是更好嗎?畢竟比直接調(diào)用還是差了一些,要是能直接調(diào)用屬性的取值方法就好了。

在C#中,可讀屬性都有一個對應(yīng)的get_XXX()的方法,可以通過調(diào)用這個方法來取得對應(yīng)屬性的值。可以使用System.Delegate.CreateDelegate創(chuàng)建一個委托來調(diào)用這個方法。

  • 通過委托調(diào)用方法來取得屬性值

我們定義一個MemberGetDelegate的委托,然后通過它來調(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);
}

然后我們測試一下,大概花了38ms,性能幾乎與直接調(diào)用一致:

利用C#獲取對象屬性值的方法

最后做一個簡單的封裝,緩存一下創(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());
  }
}

這樣使用起來就方便多了

People people = new People { Name = "Wayne" };
PropertyValue<People> propertyValue = new PropertyValue<People>(people);
object value = propertyValue.Get("Name");

看完上述內(nèi)容是否對您有幫助呢?如果還想對相關(guān)知識有進一步的了解或閱讀更多相關(guān)文章,請關(guān)注億速云行業(yè)資訊頻道,感謝您對億速云的支持。

向AI問一下細節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI