using System; using System.Data; using System.Reflection; using System.Reflection.Emit; namespace Masuit.Tools.Database { internal sealed class DataTableBuilder { private static readonly MethodInfo GetValueMethod = typeof(DataRow).GetMethod("get_Item", new[] { typeof(int) }); private static readonly MethodInfo IsDbNullMethod = typeof(DataRow).GetMethod("IsNull", new[] { typeof(int) }); private delegate T Load(DataRow dataRecord); private Load _handler; private DataTableBuilder() { } public T Build(DataRow dataRecord) { return _handler(dataRecord); } public static DataTableBuilder CreateBuilder(DataRow dataRecord) { var methodCreateEntity = new DynamicMethod("DynamicCreateEntity", typeof(T), new[] { typeof(DataRow) }, typeof(T), true); var generator = methodCreateEntity.GetILGenerator(); var result = generator.DeclareLocal(typeof(T)); generator.Emit(OpCodes.Newobj, typeof(T).GetConstructor(Type.EmptyTypes)); generator.Emit(OpCodes.Stloc, result); for (int i = 0; i < dataRecord.ItemArray.Length; i++) { var propertyInfo = typeof(T).GetProperty(dataRecord.Table.Columns[i].ColumnName); var endIfLabel = generator.DefineLabel(); if (propertyInfo == null || propertyInfo.GetSetMethod() == null) { continue; } generator.Emit(OpCodes.Ldarg_0); generator.Emit(OpCodes.Ldc_I4, i); generator.Emit(OpCodes.Callvirt, IsDbNullMethod); generator.Emit(OpCodes.Brtrue, endIfLabel); generator.Emit(OpCodes.Ldloc, result); generator.Emit(OpCodes.Ldarg_0); generator.Emit(OpCodes.Ldc_I4, i); generator.Emit(OpCodes.Callvirt, GetValueMethod); generator.Emit(OpCodes.Unbox_Any, propertyInfo.PropertyType); generator.Emit(OpCodes.Callvirt, propertyInfo.GetSetMethod()); generator.MarkLabel(endIfLabel); } generator.Emit(OpCodes.Ldloc, result); generator.Emit(OpCodes.Ret); return new DataTableBuilder { _handler = (Load)methodCreateEntity.CreateDelegate(typeof(Load)) }; } } }