4

property.setvalue() 有哪些替代方法?我读过它非常慢。我正在使用它将 IDataReader 映射到 POCO 对象。

这是代码的截断版本。这里的一切对我来说都很新鲜。我知道有很多框架可以完成这项任务。但是,我们不能使用它们。

public class DbAutoMapper<T>
{
    public IEnumerable<T> MapToList(IDataReader reader)
    {
        var list = new List<T>();

        while (reader.Read())
        {                
            var obj = Activator.CreateInstance<T>();

            foreach (PropertyInfo prop in obj.GetType().GetProperties())
            {
                foreach (var attribute in prop.GetCustomAttributes(true))
                {                        
                    prop.SetValue(obj, value, null);                        
                }
            }

            list.Add(obj);
        }
        return list;
    }
}
4

2 回答 2

5

首先:当你不使用属性时,为什么要为每个属性重复反射?

第二:假设您打算按名称、列到属性(这不是代码当前所做的)来映射它,请考虑像dapper这样的工具,它会为您完成所有这些工作,包括缓存的高性能反射发射。它还将为您处理命令本身。例如:

string region = "North";
var customers = conn.Query<Customer>(
    @"select * from Customers where Region = @region",
    new { region } // full parameterization, the easy way
).ToList();

如果您需要更多控制,请考虑FastMember,它提供快速成员访问(同样,反射发射),但不特定于数据访问:

var accessor = TypeAccessor.Create(typeof(T)); 
string propName = // something known only at runtime 
while( /* some loop of data */ ) {
    var obj = new T();
    foreach(var col in cols) {
        string propName = // ...
        object cellValue = // ...
        accessor[obj, propName] = cellValue;
    }
    yield return obj;
}
于 2012-09-12T20:10:02.463 回答
1

想到了几种方法...

跳过反射

public class DbAutoMapper<T> where T : IInitFromReader, new()
{
    public IEnumerable<T> MapToList(IDataReader reader)
    {
        var list = new List<T>();

        while (reader.Read())
        {                
            IInitFromReader obj = new T;

            obj.InitFromReader(reader);

            list.Add(obj);
        }
        return list;
    }
}

然后,您必须在每个实体对象中实现 InitFromReader。显然,这跳过了反射的好处(更少的代码)。

代码生成

为 (InitFromReader) 维护此代码很痛苦,因此您可以选择生成它。这在许多方面为您提供了两全其美:

  • 您不必(手动)维护大量代码
  • 您不会受到反射的性能影响。
于 2012-09-12T19:56:25.800 回答