亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > C# > 正文

利用lambda表達式樹優化反射詳解

2019-10-29 19:41:08
字體:
來源:轉載
供稿:網友

前言

本節重點不講反射機制,而是講lambda表達式樹來替代反射中常用的獲取屬性和方法,來達到相同的效果但卻比反射高效。

每個人都知道,用反射調用一個方法或者對屬性執行SetValue和GetValue操作的時候都會比直接調用慢很多,這其中設計到CLR中內部的處理,不做深究。然而,我們在某些情況下又無法不使用反射,比如:在一個ORM框架中,你要將一個DataRow轉化為一個對象,但你又不清楚該對象有什么屬性,這時候你就需要寫一個通用的泛型方法來處理,以下代碼寫得有點惡心,但不妨礙理解意思:

//將DataReader轉化為一個對象     private static T GetObj<T>(SqliteDataReader reader) where T : class {  T obj = new T();  PropertyInfo[] pros = obj.GetType().GetProperties();  foreach (PropertyInfo item in pros)  {  try  {   Int32 Index = reader.GetOrdinal(item.Name);   String result = reader.GetString(Index);   if (typeof(String) == item.PropertyType)   {   item.SetValue(obj, result);   continue;   }   if (typeof(DateTime) == item.PropertyType)   {   item.SetValue(obj, Convert.ToDateTime(result));   continue;   }   if (typeof(Boolean) == item.PropertyType)   {   item.SetValue(obj, Convert.ToBoolean(result));   continue;   }   if (typeof(Int32) == item.PropertyType)   {   item.SetValue(obj, Convert.ToInt32(result));   continue;   }   if (typeof(Single) == item.PropertyType)   {   item.SetValue(obj, Convert.ToSingle(result));   continue;   }   if (typeof(Single) == item.PropertyType)   {   item.SetValue(obj, Convert.ToSingle(result));   continue;   }   if (typeof(Double) == item.PropertyType)   {   item.SetValue(obj, Convert.ToDouble(result));   continue;   }   if (typeof(Decimal) == item.PropertyType)   {   item.SetValue(obj, Convert.ToDecimal(result));   continue;   }   if (typeof(Byte) == item.PropertyType)   {   item.SetValue(obj, Convert.ToByte(result));   continue;   }  }  catch (ArgumentOutOfRangeException ex)  {   continue;  }  }  return obj; }

對于這種情況,其執行效率是特別低下的,具體多慢在下面例子會在.Net Core平臺上和.Net Framework4.0運行測試案例.對于以上我舉例的情況,效率上我們還可以得到提升。但對于想在運行時修改一下屬性的名稱或其他操作,反射還是一項特別的神器,因此在某些情況下反射還是無法避免的。

但是對于只是簡單的SetValue或者GetValue,包括用反射構造函數,我們可以想一個中繼的方法,那就是使用表達式樹。對于不理解表達式樹的,可以到微軟文檔查看,點擊我。表達式樹很容易通過對象模型表示表達式,因此強烈建議學習。查看以下代碼:

static void Main() {  Dog dog = new Dog();  PropertyInfo propertyInfo = dog.GetType().GetProperty(nameof(dog.Name)); //獲取對象Dog的屬性  MethodInfo SetterMethodInfo = propertyInfo.GetSetMethod(); //獲取屬性Name的set方法  ParameterExpression param = Expression.Parameter(typeof(Dog), "param");  Expression GetPropertyValueExp = Expression.Lambda(Expression.Property(param, nameof(dog.Name)), param);  Expression<Func<Dog, String>> GetPropertyValueLambda = (Expression<Func<Dog, String>>)GetPropertyValueExp;  ParameterExpression paramo = Expression.Parameter(typeof(Dog), "param");  ParameterExpression parami = Expression.Parameter(typeof(String), "newvalue");  MethodCallExpression MethodCallSetterOfProperty = Expression.Call(paramo, SetterMethodInfo, parami);  Expression SetPropertyValueExp = Expression.Lambda(MethodCallSetterOfProperty, paramo, parami);  Expression<Action<Dog, String>> SetPropertyValueLambda = (Expression<Action<Dog, String>>)SetPropertyValueExp;  //創建了屬性Name的Get方法表達式和Set方法表達式,當然只是最簡單的  Func<Dog, String> Getter = GetPropertyValueLambda.Compile();   Action<Dog, String> Setter = SetPropertyValueLambda.Compile();  Setter?.Invoke(dog, "WLJ"); //我們現在對dog這個對象的Name屬性賦值  String dogName = Getter?.Invoke(dog); //獲取屬性Name的值    Console.WriteLine(dogName);  Console.ReadKey(); } public class Dog {  public String Name { get; set; } }

以下代碼可能很難看得懂,但只要知道我們創建了屬性的Get、Set這兩個方法就行,其結果最后也能輸出狗的名字 WLJ,擁有ExpressionTree的好處是他有一個名為Compile()的方法,它創建一個代表表達式的代碼塊。現在是最有趣的部分,假設你在編譯時不知道類型(在這篇文章中包含的代碼我在不同的程序集上創建了一個類型)你仍然可以應用這種技術,我將對于常用的屬性的set,get操作進行分裝。

/// <summary>   /// 屬性類,仿造反射中的PropertyInfo /// </summary>   public class Property {  private readonly PropertyGetter getter;  private readonly PropertySetter setter;  public String Name { get; private set; }  public PropertyInfo Info { get; private set; }  public Property(PropertyInfo propertyInfo)  {   if (propertyInfo == null)    throw new NullReferenceException("屬性不能為空");   this.Name = propertyInfo.Name;   this.Info = propertyInfo;   if (this.Info.CanRead)   {    this.getter = new PropertyGetter(propertyInfo);   }   if (this.Info.CanWrite)   {    this.setter = new PropertySetter(propertyInfo);   }  }  /// <summary>     /// 獲取對象的值  /// </summary>    /// <param name="instance"></param>    /// <returns></returns>     public Object GetValue(Object instance)  {   return getter?.Invoke(instance);  }  /// <summary>     /// 賦值操作  /// </summary>    /// <param name="instance"></param>    /// <param name="value"></param>     public void SetValue(Object instance, Object value)  {   this.setter?.Invoke(instance, value);  }  private static readonly ConcurrentDictionary<Type, Core.Reflection.Property[]> securityCache = new ConcurrentDictionary<Type, Property[]>();  public static Core.Reflection.Property[] GetProperties(Type type)  {   return securityCache.GetOrAdd(type, t => t.GetProperties().Select(p => new Property(p)).ToArray());  } }  /// <summary>   /// 屬性Get操作類  /// </summary>    public class PropertyGetter  {  private readonly Func<Object, Object> funcGet;  public PropertyGetter(PropertyInfo propertyInfo) : this(propertyInfo?.DeclaringType, propertyInfo.Name)  {  }  public PropertyGetter(Type declareType, String propertyName)  {   if (declareType == null)   {    throw new ArgumentNullException(nameof(declareType));   }   if (propertyName == null)   {    throw new ArgumentNullException(nameof(propertyName));   }   this.funcGet = CreateGetValueDeleagte(declareType, propertyName);  }  //代碼核心部分     private static Func<Object, Object> CreateGetValueDeleagte(Type declareType, String propertyName)  {   // (object instance) => (object)((declaringType)instance).propertyName       var param_instance = Expression.Parameter(typeof(Object));   var body_objToType = Expression.Convert(param_instance, declareType);   var body_getTypeProperty = Expression.Property(body_objToType, propertyName);   var body_return = Expression.Convert(body_getTypeProperty, typeof(Object));   return Expression.Lambda<Func<Object, Object>>(body_return, param_instance).Compile();  }  public Object Invoke(Object instance)  {   return this.funcGet?.Invoke(instance);  } }  public class PropertySetter {  private readonly Action<Object, Object> setFunc;  public PropertySetter(PropertyInfo property)   {   if (property == null)   {    throw new ArgumentNullException(nameof(property));   }   this.setFunc = CreateSetValueDelagate(property);  }  private static Action<Object, Object> CreateSetValueDelagate(PropertyInfo property)  {   // (object instance, object value) =>    //  ((instanceType)instance).Set_XXX((propertyType)value)   //聲明方法需要的參數   var param_instance = Expression.Parameter(typeof(Object));   var param_value = Expression.Parameter(typeof(Object));   var body_instance = Expression.Convert(param_instance, property.DeclaringType);   var body_value = Expression.Convert(param_value, property.PropertyType);   var body_call = Expression.Call(body_instance, property.GetSetMethod(), body_value);   return Expression.Lambda<Action<Object, Object>>(body_call, param_instance, param_value).Compile();  }  public void Invoke(Object instance, Object value)  {   this.setFunc?.Invoke(instance, value);  } }

在將代碼應用到實例:

   Dog dog = new Dog();   PropertyInfo propertyInfo = dog.GetType().GetProperty(nameof(dog.Name));      //反射操作   propertyInfo.SetValue(dog, "WLJ");   String result = propertyInfo.GetValue(dog) as String;   Console.WriteLine(result);      //表達式樹的操作   Property property = new Property(propertyInfo);   property.SetValue(dog, "WLJ2");   String result2 = propertyInfo.GetValue(dog) as String;   Console.WriteLine(result2);

發現其實現的目的與反射一致,但效率卻有明顯的提高。

以下測試以下他們兩之間的效率。測試代碼如下:

   Student student = new Student();   PropertyInfo propertyInfo = student.GetType().GetProperty(nameof(student.Name));   Property ExpProperty = new Property(propertyInfo);   Int32 loopCount = 1000000;   CodeTimer.Initialize(); //測試環境初始化   //下面該方法個執行1000000次   CodeTimer.Time("基礎反射", loopCount, () => {     propertyInfo.SetValue(student, "Fode",null);   });   CodeTimer.Time("lambda表達式樹", loopCount, () => {    ExpProperty.SetValue(student, "Fode");   });   CodeTimer.Time("直接賦值", loopCount, () => {    student.Name = "Fode";   });   Console.ReadKey();

其.Net4.0環境下運行結果如下:

lambda,表達式樹,反射

.Net Core環境下運行結果:

lambda,表達式樹,反射

從以上結果可以知道,迭代同樣的次數反射需要183ms,而用表達式只要34ms,直接賦值需要7ms,在效率上,使用表達式這種方法有顯著的提高,您可以看到使用此技術可以完全避免使用反射時的性能損失。反射之所以效率有點低主要取決于其加載的時候時在運行期下,而表達式則在編譯期,下篇有空將會介紹用Emit技術優化反射,會比表達式略快一點。

注:對于常用對象的屬性,最好將其緩存起來,這樣效率會更高。。

代碼下載

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對VEVB武林網的支持。


注:相關教程知識閱讀請移步到c#教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲高清一区二| 欧美在线观看视频| 伊人亚洲福利一区二区三区| 国产亚洲人成网站在线观看| 亚洲成人激情在线| 91亚洲国产精品| 亚洲女人被黑人巨大进入| 在线亚洲午夜片av大片| 色爱av美腿丝袜综合粉嫩av| 91精品久久久久久久久久| 日韩精品中文字幕久久臀| 国产成人精品一区二区| 成人高清视频观看www| 国内精品久久久久久| 日韩电影中文字幕在线| 国产成人精品最新| 亚洲国产精品嫩草影院久久| 亚洲xxxx妇黄裸体| 热久久视久久精品18亚洲精品| 久久久亚洲精品视频| 夜夜嗨av色一区二区不卡| 欧美成人午夜剧场免费观看| 欧美亚洲视频一区二区| 国产精品视频久| 国产精品一区二区女厕厕| 久久这里只有精品99| 亚洲女人被黑人巨大进入| 色综合91久久精品中文字幕| 日韩美女在线播放| 久久影院中文字幕| 国产福利精品视频| 久久福利网址导航| 国产精品高潮在线| 国内精品久久影院| 日本精品va在线观看| 成人精品一区二区三区电影免费| 亚洲精品v天堂中文字幕| 久久精品成人欧美大片古装| 国产精品亚洲自拍| 欧美成人午夜影院| 日韩在线免费观看视频| 欧美精品在线免费播放| 国产成人精品国内自产拍免费看| 亚洲综合日韩中文字幕v在线| 成人美女av在线直播| 欧美成人全部免费| 亚洲精品第一页| 欧美一级视频在线观看| 亚洲成色www8888| 亚洲免费成人av电影| 久久亚洲国产精品成人av秋霞| 久久精品99久久久久久久久| 国产精品专区h在线观看| 亚洲最大中文字幕| 激情懂色av一区av二区av| 久久99精品国产99久久6尤物| 成人网页在线免费观看| 久久天天躁狠狠躁夜夜爽蜜月| 国产欧美最新羞羞视频在线观看| 91po在线观看91精品国产性色| 日本精品免费一区二区三区| 国产91成人video| 欧美特级www| 久久资源免费视频| 亚洲一区二区三区乱码aⅴ| 欧美最猛性xxxxx免费| 国产精品网址在线| 精品国产乱码久久久久酒店| 日本在线观看天堂男亚洲| 欧美夫妻性生活xx| 国产精品视频大全| 亚洲国内高清视频| 欧美另类暴力丝袜| 欧美裸体xxxx极品少妇软件| 伊人久久久久久久久久久| 日韩一区视频在线| 欧美日韩成人在线视频| 欧美日本高清视频| 在线亚洲午夜片av大片| 国产精品久久久久久久久久东京| 国产精品国产福利国产秒拍| 色噜噜亚洲精品中文字幕| 777777777亚洲妇女| 日韩影视在线观看| 色先锋资源久久综合5566| 岛国视频午夜一区免费在线观看| 亚洲wwwav| 国产精品日日摸夜夜添夜夜av| 91沈先生在线观看| 亚洲专区中文字幕| 欧美一级淫片丝袜脚交| 欧美夫妻性视频| 国产成人综合精品在线| 日韩欧美高清视频| 欧美野外猛男的大粗鳮| 日韩精品黄色网| 亚洲护士老师的毛茸茸最新章节| 亚洲精品一区二三区不卡| 国产精品偷伦视频免费观看国产| 久久久久久尹人网香蕉| 久久精品视频免费播放| 欧美午夜片在线免费观看| 国产精品久久久精品| 色综合久久久888| 亚洲欧洲在线观看| 欧美日韩另类在线| 91精品国产91久久久久久不卡| 欧美老女人xx| 日韩精品中文字幕有码专区| 欧美性猛xxx| 91精品国产乱码久久久久久蜜臀| 69国产精品成人在线播放| 国产视频精品va久久久久久| 中文字幕九色91在线| 日本高清久久天堂| 亚洲精品日韩激情在线电影| 日韩中文字幕免费看| 97色在线视频| 亚洲自拍偷拍区| 亚洲一区二区三区sesese| 在线观看欧美日韩| 国产精品久久久久999| 亚洲国产美女久久久久| 欧美精品成人91久久久久久久| 精品国产精品三级精品av网址| 久久亚洲国产精品成人av秋霞| 26uuu亚洲伊人春色| 国产91精品黑色丝袜高跟鞋| 日本一区二区三区四区视频| 国产精品一区二区3区| 久久夜色精品亚洲噜噜国产mv| 欧美精品在线网站| 成人网在线免费观看| 国产成人拍精品视频午夜网站| 日韩av在线高清| 久久人人爽国产| 国产精品日韩欧美| 色妞色视频一区二区三区四区| 久久久成人精品视频| 亚洲视频在线观看| 高清在线视频日韩欧美| 九九久久精品一区| 国产啪精品视频网站| 91在线观看免费观看| 97香蕉久久夜色精品国产| 久热精品视频在线观看| 精品成人国产在线观看男人呻吟| 久久97精品久久久久久久不卡| 精品高清美女精品国产区| 日韩在线视频播放| 最新的欧美黄色| 欧美风情在线观看| 成人免费视频xnxx.com| 成人97在线观看视频| 欧美午夜影院在线视频| 亚洲国产中文字幕在线观看| 色偷偷噜噜噜亚洲男人的天堂| 亚洲精品98久久久久久中文字幕| 深夜福利国产精品| 色综合天天综合网国产成人网| 国产日本欧美在线观看| 国产精品久久久久久久久久| 精品露脸国产偷人在视频| 欧美亚洲成人免费|