实体框架以编程方式在模型中设置所有十进制精度值

时间:2013-10-10 01:55:11

标签: c# entity-framework

这个问题的答案(下面链接)让我想知道我的问题是否可能?

Entity Framework Code First fluent API setting field properties in a for loop

有没有办法更动态地做到这一点?我有多个模型我想要这样做,并且不希望克隆我想要配置的每个实体类的onmodelcreating中的代码。可能是一个空的基类?

1 个答案:

答案 0 :(得分:2)

在EF6中,您可以使用custom conventions执行此操作。应该只是几行代码。

在EF5中,您可能必须使用反射发现您的实体类型,然后按照上面提供的链接中的说明对其进行配置。这是我提出的代码(我偷了构建用于从post you linked above访问属性的表达式的方法):

protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
    foreach (var contextProperty in typeof(Context).GetProperties())
    {
        if (contextProperty.PropertyType.IsGenericType && 
            contextProperty.PropertyType.GetGenericTypeDefinition() == typeof(DbSet<>))
        {
            var entityType = contextProperty.PropertyType.GenericTypeArguments[0];

            foreach(var decimalProperty in entityType.GetProperties().Where(p => p.PropertyType == typeof(decimal)))
            {    
                var configurePropertyMethod = 
                    GetType()
                    .GetMethod("ConfigureProperty", BindingFlags.Static | BindingFlags.NonPublic)
                    .MakeGenericMethod(entityType);
                configurePropertyMethod.Invoke(null, new object[] { modelBuilder, decimalProperty });
            }
        }
    }
}

private static void ConfigureProperty<T>(DbModelBuilder modelBuilder, PropertyInfo propertyInfo) 
    where T : class
{
    var propertyExpression = BuildLambda<T, decimal>(propertyInfo);
    modelBuilder.Entity<T>().Property(propertyExpression).HasPrecision(10, 3);
}

private static Expression<Func<T, U>> BuildLambda<T, U>(PropertyInfo property)
{
    var param = Expression.Parameter(typeof(T), "p");
    MemberExpression memberExpression = Expression.Property(param, property);
    var lambda = Expression.Lambda<Func<T, U>>(memberExpression, param);
    return lambda;
}