【问题标题】:Using EF 4.1 Inserting a parent record and multiple child records as a single "Unit of Work"使用 EF 4.1 将父记录和多个子记录作为单个“工作单元”插入
【发布时间】:2012-02-20 02:19:17
【问题描述】:

我正在练习一组代码优先的类,它们之间存在父子关系。

public class Parent
{
    public int Id ( get; set; }
    public string Name { get; set; }
    public List<Child> Children { get; set; }

    public Parent()
    {
        Children = new List<Child>();
    }
}

public class Child()
{
    public int Id ( get; set; }
    public string Name { get; set; }
}

public class MyContext : DbContext
{
    public DbSet<Parent> Parents { get; set; }
    public DbSet<Child> Children { get; set; }
}

在一个简单的控制台应用程序中,我创建了一个父项并创建了两个子记录并将它们添加到父项。如果我随后调用 SaveChanges(),则只会将第一个孩子添加到 children 表中。

var x = new MyContext();

var c1 = new Child { Name = "Alpha-Child" };
var c2 = new Child { Name = "Beta-Child" };

var p = new Parent {Name = "Alpha-Parent"};

p.Children.Add(c1);
p.Children.Add(c2);

x.Parents.Add(p);

x.SaveChanges();

这不是我想要的,所以我尝试将 then 添加到父对象和上下文对象中。

var x = new MyContext();

var c1 = new Child { Name = "Alpha-Child" };
var c2 = new Child { Name = "Beta-Child" };

var p = new Parent {Name = "Alpha-Parent"};

p.Children.Add(c1);
p.Children.Add(c2);

x.Children.Add(c1);
x.Children.Add(c2);

x.Parents.Add(p);

x.SaveChanges();

这将两个子记录都放入数据库,但只有一个(第一个)与父记录相关联。

最后但同样重要的是,我尝试在每次调用 Parent.Add() 后调用 SaveChanges()。

var x = new MyContext();

var c1 = new Child { Name = "Alpha-Child" };
var c2 = new Child { Name = "Beta-Child" };

var p = new Parent {Name = "Alpha-Parent"};

x.Parents.Add(p);

p.Children.Add(c1);
x.SaveChanges();

p.Children.Add(c2);
x.SaveChanges();

但是,这打破了我对“工作单元”模式如何工作的想法。我“应该能够调用 SaveChanges() 一次,然后我的所有更改都会发生,对吗?

那么,我做错了什么?

【问题讨论】:

    标签: c# entity-framework


    【解决方案1】:

    以下代码将保存在单次保存更改中。并且始终使用事务范围进行单批保存。

    public class Child 
    {
      public int Id ( get; set; }
      public string Name { get; set; }
      public Parent Parent {set;get;}
    }
    
    
    var x = new MyContext();
    var c1 = new Child { Name = "Alpha-Child" };
    var c2 = new Child { Name = "Beta-Child" };
    
    var p = new Parent {Name = "Alpha-Parent"};
    c1.Parent = p;
    c2.Parent = p;
    x.Parents.Add(p);
    x.SaveChanges();
    

    【讨论】:

    • 谢谢。那确实有效。但是,为什么我需要一个返回到父级的链接?在常规的 POCO 世界中,除非业务需求需要,否则我永远不会创建这种类型的关系。我总是会从基础实体开始遍历我的实体。我是否错过了一些我应该学习/牢记的模式?
    • 不,您不需要在 Chind 中为 Parent 添加属性。 Parnet.Children 属性足以满足一对多关系。
    • 只是一个更新:为了使 Children 表中的 Parent FK 成为必需,我确实需要将链接从 child 添加回 parent 并使用注释将其标记为必需。但是,我可以将属性设为私有,以便对类的用户隐藏它。
    【解决方案2】:

    我不知道您为什么要从 EntityBase 继承,因为您提到您正在使用 POCO Code First 进行测试。 无论如何,我成功地测试了以下代码

     class Program
        {
            static void Main(string[] args)
            {
                var x = new MyContext();
    
                var c1 = new Child { Name = "Alpha-Child" };
                var c2 = new Child { Name = "Beta-Child" };
    
                var p = new Parent { Name = "Alpha-Parent" };
    
                p.Children.Add(c1);
                p.Children.Add(c2);
    
                x.Parents.Add(p);
    
                x.SaveChanges(); 
    
    
                Console.Read();
            }
        }
    
       public class Parent 
       { 
            public Parent() 
            { 
                Children = new List<Child>(); 
            } 
    
           public int Id { get; set; }
            public string Name { get; set; } 
            public List<Child> Children { get; set; } 
        } 
    
        public class Child
        { 
            public int Id { get; set; } 
            public string Name { get; set; } 
        } 
    
        public class MyContext : DbContext 
        { 
            public DbSet<Parent> Parents { get; set; } 
            public DbSet<Child> Children { get; set; } 
        } 
    

    【讨论】:

    • EntityBase ...剪切和粘贴的神器。我已经修复了我的示例。
    猜你喜欢
    • 1970-01-01
    • 2011-04-03
    • 2016-02-19
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多