如何使用Linq将实体表与交叉引用表连接起来

时间:2015-02-22 00:40:16

标签: .net linq entity-framework entity entity-framework-6

让我先说一下我对Linq来说相对较新,但我似乎已经很快掌握了大部分内容。但是这个问题让我很难过。我搜索并搜索无济于事。

我正在使用代码优先并创建了2个模型,"项目"和"关键字"。他们之间有着多对多的关系。我的实体模型看起来像这样:

public class MyContext : DbContext
{
    public MyContext() : base("name=DefaultDatabase") { }

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

        // Mapping Xref table to the correct DB table
        modelBuilder.Entity<Item>()
          .HasMany(i => i.Keywords)
          .WithMany(k => k.Items)
          .Map(ik =>
          {
              ik.ToTable("ItemKeywords", "MySchema");
              ik.MapLeftKey("ItemId");
              ik.MapRightKey("KeywordId");
          });

    }

    public virtual DbSet<Item> Items { get; set; }
    public virtual DbSet<Keyword> Keywords { get; set; }

}

public class Item
{

    public Item()
    {
        Keywords = new List<Keyword>();
    }

    public int ItemId { get; set; }
    public string Text { get; set; }
    public virtual List<Keyword> Keywords { get; set; }

}

public class Keyword
{

    public Keyword()
    {
        Items = new HashSet<Item>();
    }

    public int KeywordId { get; set; }
    public string Name { get; set; }
    public virtual ICollection<Item> Items { get; set; }

}

据我所知,这非常直截了当。当我的迁移生成数据库时,我得到3个表,&#34;项目&#34;,&#34;关键字&#34;,以及&#34; ItemKeywords&#34;的交叉引用表。 (我在&#34; OnModelCreating&#34;方法中映射)。

我遇到的问题是创建Linq查询(查询语法或方法语法,我不在乎)重现这一点:

SELECT DISTINCT
    i.*
FROM
    Items i
    INNER JOIN ItemKeywords ik
        ON i.ItemId = ik.ItemId
    INNER JOIN Keywords k
        ON ik.KeywordId = k.KeywordId
WHERE
    k.KeywordId IN (1, 3, 5, 7, 9)

所以,基本上,我想要一个独特的项目列表,其中包含一个指定的关键字。

我认为我可以通过创建一个ItemKeyword模型来实现它,但这似乎会破坏代码优先。我正在添加一个额外的模型,只是为了能够查询它。

有没有办法让这个工作没有添加第三个外部参照模型?

谢谢!

1 个答案:

答案 0 :(得分:2)

我认为你要找的是这样的查询:

 var keywordIds = new List<int> {1, 3, 5, 7, 9};
 var items =   (from s in db.Items
                from c in s.Keywords
                where keywordIds.Contains(c.KeywordId)
                select s).Distinct();

如果要将所有选定项目都带入内存,请在此查询结束时调用ToList方法。