在EF4.3中的代码优先方法中为列设置小数(16,3)

Ali*_*ghi 41 entity-framework sql-types ef-code-first

我怎样才能做到这一点 :

private decimal _SnachCount;
[Required]
[DataType("decimal(16 ,3")]
public decimal SnachCount
{
    get { return _SnachCount; }
    set { _SnachCount = value; }
}

private decimal _MinimumStock;
[Required]
[DataType("decimal(16 ,3")]
public decimal MinimumStock
{
    get { return _MinimumStock; }
    set { _MinimumStock = value; }
}

private decimal _MaximumStock;
[Required]
[DataType("decimal(16 ,3")]
public decimal MaximumStock
{
    get { return _MaximumStock; }
    set { _MaximumStock = value; }
}
Run Code Online (Sandbox Code Playgroud)

在我的模型的这一部分生成数据库后,这三列类型是十进制(18,2),为什么?这个代码错误是什么?我怎样才能做到这一点 ?

dkn*_*ack 75

DataType属性是一个验证属性.您需要使用ModelBuilder执行此操作.

public class MyContext : DbContext
{
    public DbSet<MyClass> MyClass;
    protected override void OnModelCreating(DbModelBuilder modelBuilder)
    {
        modelBuilder.Entity<MyClass>().Property(x => x.SnachCount).HasPrecision(16, 3);
        modelBuilder.Entity<MyClass>().Property(x => x.MinimumStock).HasPrecision(16, 3);
        modelBuilder.Entity<MyClass>().Property(x => x.MaximumStock).HasPrecision(16, 3);
    }
}
Run Code Online (Sandbox Code Playgroud)

  • 乐意效劳!祝你今天过得愉快! (2认同)

Rob*_*ert 32

您可以修改数据库中的所有小数标准.在您的DBContext方法OnModelCreating add line中:

modelBuilder.Properties<decimal>().Configure(c => c.HasPrecision(18, 3));
Run Code Online (Sandbox Code Playgroud)


Kin*_*rUY 21

这是从我在这里发布到同一问题的答案中复制的; /sf/answers/1077081841/.


我很高兴为此创建自定义属性:

[AttributeUsage(AttributeTargets.Property, Inherited = false, AllowMultiple = false)]
public sealed class DecimalPrecisionAttribute : Attribute
{
    public DecimalPrecisionAttribute(byte precision, byte scale)
    {
        Precision = precision;
        Scale = scale;

    }

    public byte Precision { get; set; }
    public byte Scale { get; set; }

}
Run Code Online (Sandbox Code Playgroud)

像这样使用它

[DecimalPrecision(20,10)]
public Nullable<decimal> DeliveryPrice { get; set; }
Run Code Online (Sandbox Code Playgroud)

并且通过一些反思在模型创造中发生了魔力

protected override void OnModelCreating(System.Data.Entity.ModelConfiguration.ModelBuilder modelBuilder)
{
    foreach (Type classType in from t in Assembly.GetAssembly(typeof(DecimalPrecisionAttribute)).GetTypes()
                                   where t.IsClass && t.Namespace == "YOURMODELNAMESPACE"
                                   select t)
     {
         foreach (var propAttr in classType.GetProperties(BindingFlags.Public | BindingFlags.Instance).Where(p => p.GetCustomAttribute<DecimalPrecisionAttribute>() != null).Select(
                p => new { prop = p, attr = p.GetCustomAttribute<DecimalPrecisionAttribute>(true) }))
         {

             var entityConfig = modelBuilder.GetType().GetMethod("Entity").MakeGenericMethod(classType).Invoke(modelBuilder, null);
             ParameterExpression param = ParameterExpression.Parameter(classType, "c");
             Expression property = Expression.Property(param, propAttr.prop.Name);
             LambdaExpression lambdaExpression = Expression.Lambda(property, true,
                                                                      new ParameterExpression[]
                                                                          {param});
             DecimalPropertyConfiguration decimalConfig;
             if (propAttr.prop.PropertyType.IsGenericType && propAttr.prop.PropertyType.GetGenericTypeDefinition() == typeof(Nullable<>))
             {
                 MethodInfo methodInfo = entityConfig.GetType().GetMethods().Where(p => p.Name == "Property").ToList()[7];
                 decimalConfig = methodInfo.Invoke(entityConfig, new[] { lambdaExpression }) as DecimalPropertyConfiguration;
             }
             else
             {
                 MethodInfo methodInfo = entityConfig.GetType().GetMethods().Where(p => p.Name == "Property").ToList()[6];
                 decimalConfig = methodInfo.Invoke(entityConfig, new[] { lambdaExpression }) as DecimalPropertyConfiguration;
             }

             decimalConfig.HasPrecision(propAttr.attr.Precision, propAttr.attr.Scale);
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

第一部分是获取模型中的所有类(我的自定义属性在该程序集中定义,所以我用它来获取模型的程序集)

第二个foreach使用自定义属性和属性本身获取该类中的所有属性,以便我可以获得精度和比例数据

之后我必须打电话

modelBuilder.Entity<MODEL_CLASS>().Property(c=> c.PROPERTY_NAME).HasPrecision(PRECITION,SCALE);
Run Code Online (Sandbox Code Playgroud)

所以我通过反射调用modelBuilder.Entity()并将其存储在entityConfig变量中然后构建"c => c.PROPERTY_NAME"lambda表达式

之后,如果小数可以为空,我会调用

Property(Expression<Func<TStructuralType, decimal?>> propertyExpression) 
Run Code Online (Sandbox Code Playgroud)

方法(我通过数组中的位置调用它,它不是理想我知道,任何帮助将不胜感激)

如果它不可空,我会打电话给

Property(Expression<Func<TStructuralType, decimal>> propertyExpression)
Run Code Online (Sandbox Code Playgroud)

方法.

使用DecimalPropertyConfiguration我调用HasPrecision方法.


Vul*_*sin 6

所以,我为我工作的是:

public class RestaurantItemEntity : BaseEntity
{
    [Column(TypeName = "VARCHAR(128)")]
    [StringLength(128)]
    [Required]
    public string Name { get; set; }


    [Column(TypeName = "VARCHAR(1024)")]
    [StringLength(1024)]
    public string Description { get; set; }


    [Column(TypeName = "decimal(16,2)")]
    [Required]
    public decimal Price { get; set; }


    [Required]
    public RestaurantEntity Restaurant { get; set; }
}
Run Code Online (Sandbox Code Playgroud)

这是.NET核心的EF代码第一个。