17

我在我的 DBContext 中使用了“HasConversion”来定义一个 JSonArray(语言/值)并将其保存为一个文本字段,它的作用就像一个魅力,我在我的解决方案中添加了一个新项目,没有任何改变,但后来我得到了添加有关“设置值比较器”的迁移的新错误。

我的模型就像:

    public class Brand
    {
        public int Id { get; set; }
        public new IList<LangValue> Name { get; set; } = new List<LangValue>();
    }

DBContext就像:

    modelBuilder.Entity<Brand>(t =>
    {

        t.Property(p => p.Name).HasConversion(
            v => JsonConvert.SerializeObject(v, Formatting.Indented, new JsonSerializerSettings { NullValueHandling = NullValueHandling.Include}),
            v => JsonConvert.DeserializeObject<IList<LangValue>>(v, new JsonSerializerSettings { NullValueHandling = NullValueHandling.Include})
         );
    });

它工作得很好,但是在添加一个新项目后,我在添加迁移时遇到了黄色错误,并且模型没有添加到新数据库中。

Microsoft.EntityFrameworkCore.Model.Validation[10620] 实体类型“品牌”上的属性“名称”是具有值转换器但没有值比较器的集合或枚举类型。设置一个值比较器以确保正确比较集合/枚举元素。

4

2 回答 2

15

ValueComparer 文档的解释https://docs.microsoft.com/en-us/ef/core/modeling/value-comparers#mutable-classes

列表属性的典型值转换可能会将列表与 JSON 相互转换:

modelBuilder
    .Entity<EntityType>()
    .Property(e => e.MyProperty)
    .HasConversion(
        v => JsonSerializer.Serialize(v, null),
        v => JsonSerializer.Deserialize<List<int>>(v, null));

然后,这需要ValueComparer<T>在属性上设置 a 以强制 EF Core 使用与此转换的正确比较:

var valueComparer = new ValueComparer<List<int>>(
    (c1, c2) => c1.SequenceEqual(c2),
    c => c.Aggregate(0, (a, v) => HashCode.Combine(a, v.GetHashCode())),
    c => c.ToList());

modelBuilder
    .Entity<EntityType>()
    .Property(e => e.MyProperty)
    .Metadata
    .SetValueComparer(valueComparer);
于 2020-10-13T16:37:19.490 回答
7

来自ValueComparer 类文档

为无法与 Equals(Object, Object) 比较和/或在拍摄快照时需要深度/结构副本的 CLR 类型指定自定义值快照和比较。例如,如果要检测突变,原始类型的数组将需要两者。

快照是将值的副本创建到快照中的过程,以便以后可以对其进行比较以确定它是否已更改。对于某些类型,例如集合,这需要是集合的深层副本,而不仅仅是引用的浅层副本。

您可以找到有关如何在此问题上设置 ValueComparer 的更多信息:

https://github.com/dotnet/efcore/issues/17471

于 2020-01-24T12:35:17.923 回答