5

我目前正在通过 Code First 方法学习实体框架 (EF)。我想在新创建的数据库(即由 EF 创建)上建模Professors 和es之间的简单一对多关系。Class

在我的模型中,aProfessor有一个Classes 的集合。AClass总是有一个Professor与之关联的(即相应的属性永远不会为空)。我可以创建和存储 aClass和关联的Professor,但是当我访问从数据库中检索到Classes的 a 的属性时Professor,它是空的。然而,从 a 导航Class到它的是可能的。Professor

考虑以下模型:

class Class
{
    public int Id { get; set; }
    public string Title { get; set; }
    public virtual Professor Professor { get; set; }
}

class Professor
{
    public int Id { get; set; }
    public string Name { get; set; }
    private ICollection<Class> classes = null;

    public virtual ICollection<Class> Classes
    {
        get { return classes ?? (classes = new HashSet<Class>()); }
        set { classes = value; }
    }
}

class ClassManager : DbContext
{
    public DbSet<Professor> Professors { get; set; }
    public DbSet<Class> Classes { get; set; }

    protected override void OnModelCreating(DbModelBuilder modelBuilder)
    {
        modelBuilder.Entity<Class>().HasRequired(c => c.Professor);
    }
}

在我的主要方法中,我创建和访问我的数据,如下所示:

    using (ClassManager db = new ClassManager())
    {
        Professor p = new Professor() { Name = "McTest" };
        Class c = new Class() { Title = "Testing Tests", Professor = p };

        p.Classes.Add(c);

        db.Professors.Add(p);
        db.Classes.Add(c);

        Console.WriteLine(db.SaveChanges() + " change(s) saved.");
    }

    using (ClassManager db = new ClassManager())
    {
        foreach (Professor p in db.Professors)
            Console.WriteLine("Prof. " + p.Name + " gives " + p.Classes.Count 
                + " classes.");

        foreach (Class c in db.Classes)
            Console.WriteLine(c.Title + " (Prof. " + c.Professor.Name + ")");
    }

输出是:

3 change(s) saved.
Prof. McTest gives 0 classes.
Testing Tests (Prof. McTest)

生成的数据库模式是:

table Classes
    column Id (PK, int, not null)
    column Title (nvchar(max), null)
    column Professor_Id (FK, int, not null)
table Professors
    column Id (PK, int, not null)
    column Name (nvchar(max), null)

如您所见,Classes集合属性不会出现在架构中的任何位置。这是故意的吗?

有人可以对此有所了解吗?我发现的所有示例都不会详细介绍集合属性。

4

2 回答 2

10

好的,经过数小时的搜索,我找到了解决方案。在查询数据库时,似乎必须明确指示 EF 加载实体的集合属性:

foreach (Professor p in db.Professors.Include("Classes"))
    Console.WriteLine("Prof. " + p.Name + " gives " + p.Classes.Count
        + " classes.");

Include方法就是这样做的。

于 2012-12-06T16:44:47.840 回答
1

我的主代码中两个分隔的 using 语句是故意的。如果我将它们合并在一起,Classes 属性突然不为空,但我不明白为什么。

如果 2 合并为 1,那么您“查询”存储在内存中的本地实体,而不是数据库。

在您的第一个上下文中,您不需要将两者都添加professorclass上下文中。只需添加教授,class也将被保存。

检查数据库模式并确保实体正确映射到数据库。

最后,将Professor课程更改为

class Professor
{
    public Professor()
    {
         Classes = new List<Class>();
    }

    public int Id { get; set; }
    public string Name { get; set; }
    public virtual ICollection<Class> Classes {get; set;}
}

我怀疑延迟加载被 is null 逻辑所取代,以返回一个散列集。

于 2012-12-06T13:37:57.577 回答