4

我与这 3 个表有典型的多对多关系

[Post] (
  [PostId] int, (PK)
  [Content] nvarchar(max)
   ...
)

[Tag] (
  [TagId] int, (PK)
  [Name] nvarchar
  ...
)

[TagPost] (
  [TagId] int, (PK, FK)
  [PostId] int (PK, FK)
)

而且,TagId 和 PostId 是表上相应设置的 PK 和 FK 等。然后我在 c# 中有这些类和映射

public class Post {
    public Post()
    {
        this.Tags = new HashSet<Tag>();
    }

    [Key]
    public int PostId { get; set; }
    ...

    public virtual ICollection<Tag> Tags { get; private set; }
}

public class Tag {
    public Tag()
    {
        this.Posts = new HashSet<Post>();
    }

    [Key]
    public int TagId { get; set; }
    ...

    public virtual ICollection<Post> Posts { get; private set; }
}

internal class MyDbContext : DbContext
{
    public DbSet<Post> Posts { get; set; }
    public DbSet<Tag> Tags { get; set; }

    protected override void OnModelCreating(DbModelBuilder modelBuilder)
    {
        base.OnModelCreating(modelBuilder);

        modelBuilder.Entity<Post>().ToTable("Post");
        modelBuilder.Entity<Tag>().ToTable("Tag");

        modelBuilder.Entity<Post>()
            .HasMany(x => x.Tags)
            .WithMany(x => x.Posts)
        .Map(x =>
        {
            x.ToTable("TagPost");
            x.MapLeftKey("PostId");
            x.MapRightKey("TagId");
        });
    }

然后我有这段代码来查询它们

var list = (from p in ctx.Posts.Include(p => p.Tags)
            from t in p.Tags
            where ... // some of my filter conditions
            select p).ToList();

这个加入确实返回了我正在寻找的帖子,但是即使我在那里有包含,返回的帖子也没有填写它们的关联标签。有人可以帮助指出我缺少的东西,以便我可以让标签也随帖子一起返回吗?

非常感谢。

4

2 回答 2

3

doublefrom是一个手册Join,它会导致Include被忽略,如herehere所述。Include对于其他 LINQ 方法(如 grouping 和 projections )也被忽略。

关系修复通常不适用于多对多关系,仅适用于在一端具有至少一个单一引用的关系 - 一对多或一对一。如果将Postsand 关联Tags到另一种类型(匿名或命名),则数据将正确加载,但由于关系是多对多的,EF 不会自动在内存中创建关系,因此post.Tags集合将保持为空。

要获得Include工作,您必须从查询中删除第二个from并将where子句直接应用于Post实体参数,例如:

var list = (from p in ctx.Posts.Include(p => p.Tags)
            where p.Tags.Any(t => t.TagId == 1)
            select p).ToList();

属性过滤器Tag在传递给.Any它的表达式中指定,该表达式是以Tag( t) 作为参数的表达式。

于 2013-03-22T18:04:11.513 回答
0

尝试将所有内容选择到匿名对象中(像这样)

var list = (
        from p in ctx.Posts
        from t in p.Tags
        where ... // some of my filter conditions
        select new {
            Posts = p,
            Tags = p.Tags
        })
    .ToList();

根据对我最初答案的反馈以及 EF 可以找到相关实体但无法填充Tags集合的事实,我认为问题在于 Post 类中 Tags 实体的定义。

尝试从构造函数和set 声明中删除Hashset<>初始化程序:private

public virtual ICollection<Tag> Tags { get; set; }
于 2013-03-21T19:29:48.330 回答