c# - 在多对多 Entity Framework 关系中保留外键列表

标签 c# entity-framework entity-framework-5

我的代码优先 Entity Framework 模型中有一个多对多关系。想象一下,我们有两个表,“公司”和“文章”,它们之间有这样的关系。我的简化代码模型如下所示:

public class Article
{
    public int Id { get; set; }
    public string Text { get; set; }
    public virtual ICollection<Company> Companies { get; set; }
}

public class Company
{
    public int Id { get; set; }
    public string Name { get; set; }
    public virtual ICollection<Article> Articles { get; set; }
}

使用流畅的映射,我创建了多对多关系:

modelBuilder.Entity<Company>().HasMany(c => c.Articles).WithMany(a => a.Companies);

这很好用,EF 创建了中间表。但是,根据我的逻辑,如果每个实体都有一组外键,那就太好了。这意味着我想将以下属性添加到相应的模型类中:

public virtual ICollection<int> ArticlesId { get; set; } // to Company
public virtual ICollection<int> CompaniesId { get; set; } // to Article

我知道一种解决方法是在 EF 中创建中间表的模型并在每次调用中手动选择适当的 ID,但也许 EF 映射可以提供更方便的方法来执行此类操作?预先感谢您提供任何提示。

最佳答案

不幸的是,似乎没有办法按照我想要的方式映射 ID。但是,以下是如何实现所需实体键的检索的三种变通方法。

第一个解决方案,由 Ben Reich 建议。
实现 get-only 属性,它将只返回链接实体的 ID。

public class Company
{
    public virtual ICollection<Article> Articles { get; set; }

    public IEnumerable<int> ArticlesIds
    {
        get { return Articles.Select(a => a.Id); }
    }
}

使用起来似乎很方便,但它有一个缺点——整个实体将从数据库中读取以接收唯一的 ID。以下是来自 SQL Profiler 的此类调用的日志:

exec sp_executesql N'SELECT 
[Extent2].[Id] AS [Id], 
[Extent2].[Header] AS [Header], 
[Extent2].[Description] AS [Description], 
[Extent2].[Text] AS [Text], 
[Extent2].[CreationDate] AS [CreationDate], 
[Extent2].[AccountId] AS [AccountId], 
[Extent2].[ImageSetId] AS [ImageSetId]
FROM  [dbo].[CompanyArticles] AS [Extent1]
INNER JOIN [dbo].[Articles] AS [Extent2] ON [Extent1].[Article_Id] = [Extent2].[Id]
WHERE [Extent1].[Company_Id] = @EntityKeyValue1',N'@EntityKeyValue1 int',@EntityKeyValue1=1

第二种解决方案。
使用相同的模型,在读取实体后分别读取 ID。

var ids = db.Set<Article>().Where(a => a.Companies.Select(c => c.Id).Contains(f.Id)).ToList();

这种方法与前一种方法完全相同,将获取整个实体集。

exec sp_executesql N'SELECT 
[Extent1].[Id] AS [Id], 
[Extent1].[Header] AS [Header], 
[Extent1].[Description] AS [Description], 
[Extent1].[Text] AS [Text], 
[Extent1].[CreationDate] AS [CreationDate], 
[Extent1].[AccountId] AS [AccountId], 
[Extent1].[ImageSetId] AS [ImageSetId]
FROM [dbo].[Articles] AS [Extent1]
WHERE  EXISTS (SELECT 
    1 AS [C1]
    FROM [dbo].[ArticleCompanies] AS [Extent2]
    WHERE ([Extent1].[Id] = [Extent2].[Article_Id]) AND ([Extent2].[Company_Id] = @p__linq__0)
)',N'@p__linq__0 int',@p__linq__0=1

第三种解决方案。从我的角度来看,最合适。
为您的中间表创建实体类。

public class ArticleCompany
{
    public int CompanyId { get; set; }
    public int ArticleId { get; set; }

    public virtual Company Company { get; set; }
    public virtual Article Article { get; set; }
}

将两个实体与此实体映射为一对多的关系。不要忘记映射新实体本身。

modelBuilder.Entity<Article>().HasMany(a => a.ArticlesCompanies).WithRequired(ac => ac.Article).HasForeignKey(ac => ac.ArticleId);
modelBuilder.Entity<Company>().HasMany(c => c.ArticlesCompanies).WithRequired(ac => ac.Company).HasForeignKey(ac => ac.CompanyId);

modelBuilder.Entity<ArticleCompany>().ToTable("ArticlesCompanies");
modelBuilder.Entity<ArticleCompany>().HasKey(ac => new { ac.ArticleId, ac.CompanyId });

然后,在获取实体后,使用中间表获取相关 ID:

var ids = db.Set<ArticleCompany>().Where(ca => ca.CompanyId == companyEntity.Id).Select(ca => ca.ArticleId);

对应的SQL日志(只从数据库中获取ID):

exec sp_executesql N'SELECT 
[Extent1].[ArticleId] AS [ArticleId]
FROM [dbo].[ArticlesCompanies] AS [Extent1]
WHERE [Extent1].[CompanyId] = @p__linq__0',N'@p__linq__0 int',@p__linq__0=1

关于c# - 在多对多 Entity Framework 关系中保留外键列表,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18718740/

相关文章:

c# - DataGridView AllowUserToAddRow 属性不起作用

c# - 如果 Entity Framework/DbContext 是 DAL/Repository,它在 3 层架构中的什么位置?

c# - 无法转换类型为 'System.Collections.Generic.List` 的对象 1[ModelA ]' to type ' ModelA'

c# - AddOrUpdate 未按预期工作并产生重复项

c# - 在 C# 中如何对 XML 进行编码以在页面的 JavaScript 部分将其输出到 JSON 中

c# - RSS 提要中 pubDate 的 DateTime.Parse

c# - 如何在 C# 中编写条件锁?

C#异步LCD写入

c# - Entity Framework Code First TPH 继承 - 不同的子类可以共享一个字段吗?

c# - Entity Framework DbContext 被处置