C# 類對映的四種方法【解決硬編碼的問題】(工具三)
方法一:泛型快取+表示式目錄樹實現Mapper
public class ExpressionGenericMapper<TIn, TOut>//Mapper`2 //正對於每兩個不同型別的組合都會生成副本
{
private static Func<TIn, TOut> _FUNC = null;// 在每個副本中都有一個委託
static ExpressionGenericMapper() //靜態建構函式 生成表示式目錄樹
{
ParameterExpression parameterExpression = Expression.Parameter(typeof(TIn), "p");
List<MemberBinding> memberBindingList = new List<MemberBinding>();
foreach (var item in typeof(TOut).GetProperties())
{
MemberExpression property = Expression.Property(parameterExpression, typeof(TIn).GetProperty(item.Name));
MemberBinding memberBinding = Expression.Bind(item, property);
memberBindingList.Add(memberBinding);
}
foreach (var item in typeof(TOut).GetFields())
{
MemberExpression property = Expression.Field(parameterExpression, typeof(TIn).GetField(item.Name));
MemberBinding memberBinding = Expression.Bind(item, property);
memberBindingList.Add(memberBinding);
}
MemberInitExpression memberInitExpression = Expression.MemberInit(Expression.New(typeof(TOut)), memberBindingList.ToArray());
Expression<Func<TIn, TOut>> lambda = Expression.Lambda<Func<TIn, TOut>>(memberInitExpression, new ParameterExpression[]
{
parameterExpression
});
_FUNC = lambda.Compile();//拼裝是一次性的 轉換成委託以後放入副本的靜態變數中去
}
public static TOut Trans(TIn t) // 直接獲取副本的靜態變數(委託)
{
return _FUNC(t);
}
}
方法二:序列化和反序列化實現Mapper
public class SerializeMapper
{
/// <summary>
/// 序列化反序列化方式
/// </summary>
/// <typeparam name="TIn"></typeparam>
/// <typeparam name="TOut"></typeparam>
public static TOut Trans<TIn, TOut>(TIn tIn)
{
return JsonConvert.DeserializeObject<TOut>(JsonConvert.SerializeObject(tIn));
}
}
方法三:表示式目錄樹實現Mapper
public class ExpressionMapper
{
/// <summary>
/// 字典快取--hash分佈
/// </summary>
private static Dictionary<string, object> _Dic = new Dictionary<string, object>(); //超過一定的數量之後 在字典中獲取值就會有效能損耗
/// <summary>
/// 字典快取表示式樹
/// </summary>
/// <typeparam name="TIn"></typeparam>
/// <typeparam name="TOut"></typeparam>
/// <param name="tIn"></param>
/// <returns></returns>
public static TOut Trans<TIn, TOut>(TIn tIn)
{
string key = string.Format("funckey_{0}_{1}", typeof(TIn).FullName, typeof(TOut).FullName);
if (!_Dic.ContainsKey(key))
{
//動態的生成表示式
ParameterExpression parameterExpression = Expression.Parameter(typeof(TIn), "p");
List<MemberBinding> memberBindingList = new List<MemberBinding>();
foreach (var item in typeof(TOut).GetProperties())
{
MemberExpression property = Expression.Property(parameterExpression, typeof(TIn).GetProperty(item.Name));
MemberBinding memberBinding = Expression.Bind(item, property);
memberBindingList.Add(memberBinding);
}
foreach (var item in typeof(TOut).GetFields())
{
MemberExpression property = Expression.Field(parameterExpression, typeof(TIn).GetField(item.Name));
MemberBinding memberBinding = Expression.Bind(item, property);
memberBindingList.Add(memberBinding);
}
MemberInitExpression memberInitExpression = Expression.MemberInit(Expression.New(typeof(TOut)), memberBindingList.ToArray());
Expression<Func<TIn, TOut>> lambda = Expression.Lambda<Func<TIn, TOut>>(memberInitExpression, new ParameterExpression[]
{
parameterExpression
});
Func<TIn, TOut> func = lambda.Compile();//拼裝是一次性的
_Dic[key] = func;
}
return ((Func<TIn, TOut>)_Dic[key]).Invoke(tIn);
}
}
方法四:反射實現Mapper
public static TOut Trans<TIn, TOut>(TIn tIn)
{
TOut tOut = Activator.CreateInstance<TOut>();
foreach (var itemOut in tOut.GetType().GetProperties())
{
var propIn = tIn.GetType().GetProperty(itemOut.Name);
itemOut.SetValue(tOut, propIn.GetValue(tIn));
}
foreach (var itemOut in tOut.GetType().GetFields())
{
var fieldIn = tIn.GetType().GetField(itemOut.Name);
itemOut.SetValue(tOut, fieldIn.GetValue(tIn));
}
return tOut;
}
四種方式的比較第一種方法效能最好通用性最強
相關文章
- 解決代理超時問題的三種方法
- Docker 埠對映問題解決Docker
- windows 7常見的三種字型問題的解決方法Windows
- 命令列的亂碼以及編碼的問題的解決方法命令列
- 解決java“錯誤:編碼GBK的不可對映字元”Java字元
- maven錯誤解決:編碼GBK的不可對映字元Maven字元
- Verilog程式碼和FPGA硬體的對映關係(四)FPGA
- maven的編碼問題、解決和疑問Maven
- 為硬體保留記憶體 問題的解決方法記憶體
- 問題解決方法有三
- 解決ajax跨域問題的多種方法跨域
- [疑問] [已解決] updateOrCreate () 這類方法應對併發請求的問題
- 對於一個編解碼問題的思考
- 玩轉dnmp之埠對映問題解決方案
- 資料探勘主要解決四類問題
- CentOS中文亂碼問題的解決方法CentOS
- 交叉編譯庫依賴問題的解決方法編譯
- 三種方法教你解決輸入法不顯示問題
- 開源規則引擎——ice:致力於解決靈活繁複的硬編碼問題
- 5種常見Bean對映工具的效能比對Bean
- 解決:javadoc生成出現錯誤“編碼 GBK 的不可對映字元”Java字元
- 遇到問題的解決方法
- 解決 requests 庫 URL 編碼問題
- 解決 apache tomcat 編碼問題ApacheTomcat
- Netty解決粘包和拆包問題的四種方案Netty
- MYSQL亂碼問題解決方法MySql
- 解決IP盜用問題的三種技術手段 (轉)
- 解決Make時,“/usr/bin/ld: 找不到 -lXXX”問題的三種方法
- 你解決的問題比你編寫的程式碼更重要!
- 第三章SAP Basis 解決問題的方法
- 四種方案解決ScrollView巢狀ListView問題View巢狀
- C#控制元件的閃爍問題解決方法總結C#控制元件
- 解決ASP.NET中的各種亂碼問題ASP.NET
- hibernate對映合成模式的問題模式
- 三種方法解決浮動元素父容器高度自適應問題
- Josephus問題解決方法四(迴圈陣列)陣列
- maven 使用Maven編譯專案遇到——“maven編碼gbk的不可對映字元”解決辦法Maven編譯字元
- Mysql中文亂碼問題的最佳解決方法MySql