Установите десятичное значение (16, 3) для столбца в первом подходе кода в EF4.3
Как это сделать:
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; }
}
После генерации базы данных этой частью моей модели эти три типа столбцов являются десятичными (18,2), почему?
что это за ошибка кода? как я могу это сделать?
Ответы
Ответ 1
Атрибут 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);
}
}
Ответ 2
Вы можете изменить все десятичные пропозиции в базе данных. В вашем DBC-тексте в методе OnModelCreating добавить строку:
modelBuilder.Properties<decimal>().Configure(c => c.HasPrecision(18, 3));
Ответ 3
Это скопировано из ответа, который я отправил на тот же вопрос; fooobar.com/info/37048/....
Мне было приятно создать пользовательский атрибут для этого:
[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; }
}
используя это как
[DecimalPrecision(20,10)]
public Nullable<decimal> DeliveryPrice { get; set; }
и волшебство происходит при создании модели с некоторым отражением
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);
}
}
}
первая часть состоит в том, чтобы получить все классы в модели (мой пользовательский атрибут определен в этой сборке, поэтому я использовал это, чтобы получить сборку с моделью)
второй foreach получает все свойства в этом классе с пользовательским атрибутом и сам атрибут, поэтому я могу получить данные о точности и масштабах
после этого мне нужно позвонить
modelBuilder.Entity<MODEL_CLASS>().Property(c=> c.PROPERTY_NAME).HasPrecision(PRECITION,SCALE);
поэтому я вызываю modelBuilder.Entity() отражением и сохраняю его в переменной entityConfig
затем я создаю выражение лямбда "c = > c.PROPERTY_NAME"
После этого, если десятичное значение является нулевым, я вызываю
Property(Expression<Func<TStructuralType, decimal?>> propertyExpression)
(я называю это положением в массиве, он не идеален, я знаю, любая помощь будет очень оценена)
и если он не является нулевым, я вызываю
Property(Expression<Func<TStructuralType, decimal>> propertyExpression)
метод.
Имея DecimalPropertyConfiguration, я вызываю метод HasPrecision.