使用graphdiff进行条件映射

我在DbContext有以下实体:

在这里输入图像说明

 public class A { public A() { Bs = new List<B>(); } public ICollection<B> Bs { set; get; } } 

有时我想更新图表:

 var a = dbContext.As .AsNoTracking() .Include(x=>x.Bs) .firstOrDefault(); var c = new C(); a.Bs.Add(c); var d = new D(); var e1 = new E(); var e2 = new E(); d.Es.Add(e1); //<-- added new E d.Es.Add(e2); //<-- added new E a.Bs.Add(d); 

我想使用graphdiff更新它的Bs (更新CDE ):

 dbContext.UpdateGraph(a,map=>map.OwnedCollection(x=>x.Bs)); 

这更新ABCD ,但不是E

所以我认为,我需要为graphdiff定义一个条件映射,来更新E s,比如:

 dbContext.UpdateGraph(a,map=>map.OwnedCollection(x=>x.Bs.OfType<D>(), with =>with.OwnedCollection(t=>t.Es)) .OwnedCollection(x=>x.Bs.OfType<C>())); 

有没有办法做这个工作?

你可以在graphdiff中使用它:

 dbContext.UpdateGraph(a, map => map .OwnedCollection(b => p.Bs, with => with .AssociatedCollection(p => p.Es))); 

看到这个链接: GraphDiff说明

我不相信这是可能的使用您当前的类结构。 不过,我find了一种方法来实现这一点,但是您必须对代码进行一些更改。

更新A

 public class A { public A() { Cs = new List<C>(); Ds = new List<D>(); } //PK public int AId { get; set; } public ICollection<C> Cs { set; get; } public ICollection<D> Ds { set; get; } } 

更新BCD

 public class B { public int BId { get; set; } } public class C : B { //FK that links C to A public int FK_C_AId { get; set; } } public class D : B { //FK that links D to A public int FK_D_AId { get; set; } public ICollection<E> Es { get; set; } public D() { Es = new List<E>(); } } 

为了维护TPH策略,一些映射是必要的。

 modelBuilder.Entity<A>() .HasMany(i => i.Cs) .WithRequired() .HasForeignKey(i => i.FK_C_AId) .WillCascadeOnDelete(false); modelBuilder.Entity<A>() .HasMany(i => i.Ds) .WithRequired() .HasForeignKey(i => i.FK_D_AId) .WillCascadeOnDelete(false); modelBuilder.Entity<B>() .Map<C>(m => m.Requires("Discriminator").HasValue("C")) .Map<D>(m => m.Requires("Discriminator").HasValue("D")); 

现在,你有几乎相同的数据库结构。 CD仍然映射到同一个表。

最后,更新图表:

 context.UpdateGraph(a, map => map .OwnedCollection(b => b.Cs) .OwnedCollection(b => b.Ds, with => with .AssociatedCollection(e => e.Es))); 

希望能帮助到你!