关系Relationships

关系定义两个实体之间的关系。 在关系数据库中,这由外键约束表示。

备注

本文中的大多数示例都使用一对多关系来演示概念。 有关一对一关系和多对多关系的示例,请参阅文章末尾的其他关系模式部分。

术语定义Definition of terms

有许多术语用于描述关系

  • 相关实体: 这是包含外键属性的实体。 有时称为关系的 “子级”。

  • 主体实体: 这是包含主/备用键属性的实体。 有时称为关系的 “父项”。

  • 主体密钥: 唯一标识主体实体的属性。 这可能是主键或备用密钥。

  • 外键: 用于存储相关实体的主体键值的依赖实体中的属性。

  • 导航属性: 在主体和/或从属实体上定义的属性,该属性引用相关实体。

    • 集合导航属性: 一个导航属性,其中包含对多个相关实体的引用。

    • 引用导航属性: 保存对单个相关实体的引用的导航属性。

    • 反向导航属性: 讨论特定导航属性时,此术语是指关系另一端的导航属性。

  • 自引用关系: 依赖关系和主体实体类型相同的关系。

下面的代码显示与之间的一对多关系 Blog``Post

  1. public class Blog
  2. {
  3. public int BlogId { get; set; }
  4. public string Url { get; set; }
  5. public List<Post> Posts { get; set; }
  6. }
  7. public class Post
  8. {
  9. public int PostId { get; set; }
  10. public string Title { get; set; }
  11. public string Content { get; set; }
  12. public int BlogId { get; set; }
  13. public Blog Blog { get; set; }
  14. }
  • Post是依赖实体

  • Blog是主体实体

  • Blog.BlogId是主体键(在本例中为主密钥,而不是备用密钥)

  • Post.BlogId为外键

  • Post.Blog是一个引用导航属性

  • Blog.Posts是集合导航属性

  • Post.Blog是的反向导航属性 Blog.Posts (反之亦然)

约定Conventions

默认情况下,当在某个类型上发现导航属性时,将创建一个关系。 如果当前数据库提供程序无法将其指向的类型映射为标量类型,则该属性被视为导航属性。

备注

按约定发现的关系将始终以主体实体的主键为目标。 若要以备用密钥为目标,则必须使用熟知的 API 执行其他配置。

完全定义的关系Fully defined relationships

关系最常见的模式是在关系两端定义导航属性,在依赖实体类中定义外键属性。

  • 如果在两个类型之间找到一对导航属性,则这些属性将配置为同一关系的反向导航属性。

  • 如果依赖实体包含名称与其中一种模式相匹配的属性,则该属性将被配置为外键:

    • <navigation property name><principal key property name>
    • <navigation property name>Id
    • <principal entity name><principal key property name>
    • <principal entity name>Id
  1. public class Blog
  2. {
  3. public int BlogId { get; set; }
  4. public string Url { get; set; }
  5. public List<Post> Posts { get; set; }
  6. }
  7. public class Post
  8. {
  9. public int PostId { get; set; }
  10. public string Title { get; set; }
  11. public string Content { get; set; }
  12. public int BlogId { get; set; }
  13. public Blog Blog { get; set; }
  14. }

在此示例中,突出显示的属性将用于配置关系。

备注

如果属性为主键,或者为与主体键不兼容的类型,则不会将其配置为外键。

备注

在 EF Core 3.0 之前,名为与主体键属性完全相同的属性也与外键匹配

无外键属性No foreign key property

尽管建议在依赖实体类中定义外键属性,但这并不是必需的。 如果未找到外键属性,则会使用名称引入阴影外键属性<navigation property name><principal key property name> <principal entity name><principal key property name> 如果依赖类型上没有导航,则为。

  1. public class Blog
  2. {
  3. public int BlogId { get; set; }
  4. public string Url { get; set; }
  5. public List<Post> Posts { get; set; }
  6. }
  7. public class Post
  8. {
  9. public int PostId { get; set; }
  10. public string Title { get; set; }
  11. public string Content { get; set; }
  12. public Blog Blog { get; set; }
  13. }

在此示例中,阴影外键是 BlogId 因为预先计算导航名称将是冗余的。

备注

如果已存在具有相同名称的属性,则会以数字作为后缀的阴影属性名称。

单个导航属性Single navigation property

只包含一个导航属性(无反向导航,没有外键属性)就足以具有约定定义的关系。 还可以有一个导航属性和一个外键属性。

  1. public class Blog
  2. {
  3. public int BlogId { get; set; }
  4. public string Url { get; set; }
  5. public List<Post> Posts { get; set; }
  6. }
  7. public class Post
  8. {
  9. public int PostId { get; set; }
  10. public string Title { get; set; }
  11. public string Content { get; set; }
  12. }

限制Limitations

如果有多个在两种类型之间定义的导航属性(即,不仅仅是一对相互指向的导航属性),则导航属性表示的关系是不明确的。 你将需要手动对其进行配置以解决歧义。

级联删除Cascade delete

按照约定,级联删除将对所需的关系和ClientSetNull设置为cascade ,以实现可选关系。 Cascade表示也会删除依赖实体。 ClientSetNull表示未加载到内存中的依赖实体将保持不变,必须手动删除,或将其更新为指向有效的主体实体。 对于加载到内存中的实体,EF Core 将尝试将外键属性设置为 null。

请参阅 required和 optional关系部分,了解必需和可选关系之间的差异。

有关不同的删除行为和约定使用的默认值的详细信息,请参阅级联删除

手动配置Manual configuration

若要在熟知的 API 中配置关系,请首先标识构成关系的导航属性。 HasOneHasMany 标识要开始配置的实体类型上的导航属性。 然后,将调用链接到 WithOneWithMany 以标识反向导航。 HasOne/WithOne用于引用导航属性,用于 HasMany / WithMany 集合导航属性。

  1. class MyContext : DbContext
  2. {
  3. public DbSet<Blog> Blogs { get; set; }
  4. public DbSet<Post> Posts { get; set; }
  5. protected override void OnModelCreating(ModelBuilder modelBuilder)
  6. {
  7. modelBuilder.Entity<Post>()
  8. .HasOne(p => p.Blog)
  9. .WithMany(b => b.Posts);
  10. }
  11. }
  12. public class Blog
  13. {
  14. public int BlogId { get; set; }
  15. public string Url { get; set; }
  16. public List<Post> Posts { get; set; }
  17. }
  18. public class Post
  19. {
  20. public int PostId { get; set; }
  21. public string Title { get; set; }
  22. public string Content { get; set; }
  23. public Blog Blog { get; set; }
  24. }

您可以使用数据批注来配置依赖项和主体实体上的导航属性如何配对。 这通常在两个实体类型之间存在多个导航属性对时执行。

  1. public class Post
  2. {
  3. public int PostId { get; set; }
  4. public string Title { get; set; }
  5. public string Content { get; set; }
  6. public int AuthorUserId { get; set; }
  7. public User Author { get; set; }
  8. public int ContributorUserId { get; set; }
  9. public User Contributor { get; set; }
  10. }
  11. public class User
  12. {
  13. public string UserId { get; set; }
  14. public string FirstName { get; set; }
  15. public string LastName { get; set; }
  16. [InverseProperty("Author")]
  17. public List<Post> AuthoredPosts { get; set; }
  18. [InverseProperty("Contributor")]
  19. public List<Post> ContributedToPosts { get; set; }
  20. }

备注

只能在依赖实体上的属性上使用 [Required] 来影响关系的 requiredness。 [必需] 在主体实体的导航中通常会忽略,但这可能会导致实体成为依赖实体。

备注

数据批注 [ForeignKey][InverseProperty] 在命名空间中可用 System.ComponentModel.DataAnnotations.Schema[Required]System.ComponentModel.DataAnnotations 命名空间中可用。

单个导航属性Single navigation property

如果只有一个导航属性,则和的无参数重载 WithOne WithMany 。 这表示在概念上,关系的另一端有一个引用或集合,但实体类中不包含导航属性。

  1. class MyContext : DbContext
  2. {
  3. public DbSet<Blog> Blogs { get; set; }
  4. public DbSet<Post> Posts { get; set; }
  5. protected override void OnModelCreating(ModelBuilder modelBuilder)
  6. {
  7. modelBuilder.Entity<Blog>()
  8. .HasMany(b => b.Posts)
  9. .WithOne();
  10. }
  11. }
  12. public class Blog
  13. {
  14. public int BlogId { get; set; }
  15. public string Url { get; set; }
  16. public List<Post> Posts { get; set; }
  17. }
  18. public class Post
  19. {
  20. public int PostId { get; set; }
  21. public string Title { get; set; }
  22. public string Content { get; set; }
  23. }

配置导航属性Configuring navigation properties

创建导航属性后,你可能需要对其进行进一步配置。 在 EFCore 5.0 中添加了新的流畅 API,以允许执行该配置。

  1. protected override void OnModelCreating(ModelBuilder modelBuilder)
  2. {
  3. modelBuilder.Entity<Blog>()
  4. .HasMany(b => b.Posts)
  5. .WithOne();
  6. modelBuilder.Entity<Blog>()
  7. .Navigation(b => b.Posts)
  8. .UsePropertyAccessMode(PropertyAccessMode.Property);
  9. }

备注

此调用不能用于创建导航属性。 它仅用于配置导航属性,该属性以前是通过定义关系或从约定创建的。

外键Foreign key

您可以使用熟知的 API 来配置应用作给定关系的外键属性的属性:

  1. class MyContext : DbContext
  2. {
  3. public DbSet<Blog> Blogs { get; set; }
  4. public DbSet<Post> Posts { get; set; }
  5. protected override void OnModelCreating(ModelBuilder modelBuilder)
  6. {
  7. modelBuilder.Entity<Post>()
  8. .HasOne(p => p.Blog)
  9. .WithMany(b => b.Posts)
  10. .HasForeignKey(p => p.BlogForeignKey);
  11. }
  12. }
  13. public class Blog
  14. {
  15. public int BlogId { get; set; }
  16. public string Url { get; set; }
  17. public List<Post> Posts { get; set; }
  18. }
  19. public class Post
  20. {
  21. public int PostId { get; set; }
  22. public string Title { get; set; }
  23. public string Content { get; set; }
  24. public int BlogForeignKey { get; set; }
  25. public Blog Blog { get; set; }
  26. }

您可以使用熟知的 API 来配置哪些属性应用作给定关系的复合外键属性:

  1. class MyContext : DbContext
  2. {
  3. public DbSet<Car> Cars { get; set; }
  4. protected override void OnModelCreating(ModelBuilder modelBuilder)
  5. {
  6. modelBuilder.Entity<Car>()
  7. .HasKey(c => new { c.State, c.LicensePlate });
  8. modelBuilder.Entity<RecordOfSale>()
  9. .HasOne(s => s.Car)
  10. .WithMany(c => c.SaleHistory)
  11. .HasForeignKey(s => new { s.CarState, s.CarLicensePlate });
  12. }
  13. }
  14. public class Car
  15. {
  16. public string State { get; set; }
  17. public string LicensePlate { get; set; }
  18. public string Make { get; set; }
  19. public string Model { get; set; }
  20. public List<RecordOfSale> SaleHistory { get; set; }
  21. }
  22. public class RecordOfSale
  23. {
  24. public int RecordOfSaleId { get; set; }
  25. public DateTime DateSold { get; set; }
  26. public decimal Price { get; set; }
  27. public string CarState { get; set; }
  28. public string CarLicensePlate { get; set; }
  29. public Car Car { get; set; }
  30. }

您可以使用数据批注来配置应用作给定关系的外键属性的属性。 通常,当不按约定发现外键属性时,会执行此操作:

  1. public class Blog
  2. {
  3. public int BlogId { get; set; }
  4. public string Url { get; set; }
  5. public List<Post> Posts { get; set; }
  6. }
  7. public class Post
  8. {
  9. public int PostId { get; set; }
  10. public string Title { get; set; }
  11. public string Content { get; set; }
  12. public int BlogForeignKey { get; set; }
  13. [ForeignKey("BlogForeignKey")]
  14. public Blog Blog { get; set; }
  15. }

提示

[ForeignKey]批注可放置在关系中的任一导航属性上。 它不需要在依赖实体类中定位导航属性。

备注

[ForeignKey]在导航属性上使用指定的属性不需要存在于依赖类型上。 在这种情况下,将使用指定的名称创建阴影外键。

影子外键Shadow foreign key

您可以使用的字符串重载将 HasForeignKey(...) 影子属性配置为外键(有关详细信息,请参阅影子属性)。 建议先将影子属性显式添加到模型,然后再将其用作外键(如下所示)。

  1. class MyContext : DbContext
  2. {
  3. public DbSet<Blog> Blogs { get; set; }
  4. public DbSet<Post> Posts { get; set; }
  5. protected override void OnModelCreating(ModelBuilder modelBuilder)
  6. {
  7. // Add the shadow property to the model
  8. modelBuilder.Entity<Post>()
  9. .Property<int>("BlogForeignKey");
  10. // Use the shadow property as a foreign key
  11. modelBuilder.Entity<Post>()
  12. .HasOne(p => p.Blog)
  13. .WithMany(b => b.Posts)
  14. .HasForeignKey("BlogForeignKey");
  15. }
  16. }
  17. public class Blog
  18. {
  19. public int BlogId { get; set; }
  20. public string Url { get; set; }
  21. public List<Post> Posts { get; set; }
  22. }
  23. public class Post
  24. {
  25. public int PostId { get; set; }
  26. public string Title { get; set; }
  27. public string Content { get; set; }
  28. public Blog Blog { get; set; }
  29. }

Foreign key 约束名称Foreign key constraint name

按照约定,在面向关系数据库时,外键约束命名为 FK_ 。 对于复合外键, 将成为外键属性名称的下划线分隔列表。

你还可以配置约束名称,如下所示:

  1. protected override void OnModelCreating(ModelBuilder modelBuilder)
  2. {
  3. modelBuilder.Entity<Post>()
  4. .HasOne(p => p.Blog)
  5. .WithMany(b => b.Posts)
  6. .HasForeignKey(p => p.BlogId)
  7. .HasConstraintName("ForeignKey_Post_Blog");
  8. }

无导航属性Without navigation property

不一定需要提供导航属性。 您可以直接在关系的一端提供外键。

  1. class MyContext : DbContext
  2. {
  3. public DbSet<Blog> Blogs { get; set; }
  4. public DbSet<Post> Posts { get; set; }
  5. protected override void OnModelCreating(ModelBuilder modelBuilder)
  6. {
  7. modelBuilder.Entity<Post>()
  8. .HasOne<Blog>()
  9. .WithMany()
  10. .HasForeignKey(p => p.BlogId);
  11. }
  12. }
  13. public class Blog
  14. {
  15. public int BlogId { get; set; }
  16. public string Url { get; set; }
  17. }
  18. public class Post
  19. {
  20. public int PostId { get; set; }
  21. public string Title { get; set; }
  22. public string Content { get; set; }
  23. public int BlogId { get; set; }
  24. }

主体密钥Principal key

如果你希望外键引用主键之外的属性,则可以使用熟知的 API 来配置关系的主体键属性。 配置为主体密钥的属性将自动设置为备用密钥

  1. class MyContext : DbContext
  2. {
  3. public DbSet<Car> Cars { get; set; }
  4. protected override void OnModelCreating(ModelBuilder modelBuilder)
  5. {
  6. modelBuilder.Entity<RecordOfSale>()
  7. .HasOne(s => s.Car)
  8. .WithMany(c => c.SaleHistory)
  9. .HasForeignKey(s => s.CarLicensePlate)
  10. .HasPrincipalKey(c => c.LicensePlate);
  11. }
  12. }
  13. public class Car
  14. {
  15. public int CarId { get; set; }
  16. public string LicensePlate { get; set; }
  17. public string Make { get; set; }
  18. public string Model { get; set; }
  19. public List<RecordOfSale> SaleHistory { get; set; }
  20. }
  21. public class RecordOfSale
  22. {
  23. public int RecordOfSaleId { get; set; }
  24. public DateTime DateSold { get; set; }
  25. public decimal Price { get; set; }
  26. public string CarLicensePlate { get; set; }
  27. public Car Car { get; set; }
  28. }
  1. class MyContext : DbContext
  2. {
  3. public DbSet<Car> Cars { get; set; }
  4. protected override void OnModelCreating(ModelBuilder modelBuilder)
  5. {
  6. modelBuilder.Entity<RecordOfSale>()
  7. .HasOne(s => s.Car)
  8. .WithMany(c => c.SaleHistory)
  9. .HasForeignKey(s => new { s.CarState, s.CarLicensePlate })
  10. .HasPrincipalKey(c => new { c.State, c.LicensePlate });
  11. }
  12. }
  13. public class Car
  14. {
  15. public int CarId { get; set; }
  16. public string State { get; set; }
  17. public string LicensePlate { get; set; }
  18. public string Make { get; set; }
  19. public string Model { get; set; }
  20. public List<RecordOfSale> SaleHistory { get; set; }
  21. }
  22. public class RecordOfSale
  23. {
  24. public int RecordOfSaleId { get; set; }
  25. public DateTime DateSold { get; set; }
  26. public decimal Price { get; set; }
  27. public string CarState { get; set; }
  28. public string CarLicensePlate { get; set; }
  29. public Car Car { get; set; }
  30. }

警告

指定主体键属性的顺序必须与为外键指定这些属性的顺序一致。

必需和可选的关系Required and optional relationships

您可以使用熟知的 API 来配置关系是必需的还是可选的。 最终,这会控制外键属性是必需的还是可选的。 当使用阴影状态外键时,这非常有用。 如果实体类中具有外键属性,则关系的 requiredness 取决于外键属性是必需还是可选(有关详细信息,请参阅必需和可选属性)。

  1. protected override void OnModelCreating(ModelBuilder modelBuilder)
  2. {
  3. modelBuilder.Entity<Post>()
  4. .HasOne(p => p.Blog)
  5. .WithMany(b => b.Posts)
  6. .IsRequired();
  7. }

备注

IsRequired(false)如果未配置,则调用还会使外键属性为可选。

级联删除Cascade delete

您可以使用熟知的 API 显式配置给定关系的级联删除行为。

有关每个选项的详细讨论,请参阅级联删除

  1. protected override void OnModelCreating(ModelBuilder modelBuilder)
  2. {
  3. modelBuilder.Entity<Post>()
  4. .HasOne(p => p.Blog)
  5. .WithMany(b => b.Posts)
  6. .OnDelete(DeleteBehavior.Cascade);
  7. }

其他关系模式Other relationship patterns

一对一One-to-one

一对多关系在两侧都有一个引用导航属性。 它们遵循与一对多关系相同的约定,但在外键属性上引入了唯一索引,以确保只有一个依赖项与每个主体相关。

  1. public class Blog
  2. {
  3. public int BlogId { get; set; }
  4. public string Url { get; set; }
  5. public BlogImage BlogImage { get; set; }
  6. }
  7. public class BlogImage
  8. {
  9. public int BlogImageId { get; set; }
  10. public byte[] Image { get; set; }
  11. public string Caption { get; set; }
  12. public int BlogId { get; set; }
  13. public Blog Blog { get; set; }
  14. }

备注

EF 会根据其检测外键属性的能力,选择其中一个实体作为依赖项。 如果选择了错误的实体作为依赖项,则可以使用熟知的 API 来更正此问题。

使用 “流畅” API 配置关系时,请使用 HasOneWithOne 方法。

配置外键时,需要指定依赖实体类型-请注意以下列表中提供的泛型参数 HasForeignKey 。 在一对多关系中,可以清楚地表明具有引用导航的实体是依赖项,并且具有集合的实体是主体。 但这并不是一对一的关系,因此需要显式定义它。

  1. class MyContext : DbContext
  2. {
  3. public DbSet<Blog> Blogs { get; set; }
  4. public DbSet<BlogImage> BlogImages { get; set; }
  5. protected override void OnModelCreating(ModelBuilder modelBuilder)
  6. {
  7. modelBuilder.Entity<Blog>()
  8. .HasOne(b => b.BlogImage)
  9. .WithOne(i => i.Blog)
  10. .HasForeignKey<BlogImage>(b => b.BlogForeignKey);
  11. }
  12. }
  13. public class Blog
  14. {
  15. public int BlogId { get; set; }
  16. public string Url { get; set; }
  17. public BlogImage BlogImage { get; set; }
  18. }
  19. public class BlogImage
  20. {
  21. public int BlogImageId { get; set; }
  22. public byte[] Image { get; set; }
  23. public string Caption { get; set; }
  24. public int BlogForeignKey { get; set; }
  25. public Blog Blog { get; set; }
  26. }

多对多Many-to-many

目前尚不支持多对多关系,没有实体类来表示联接表。 但是,您可以通过包含联接表的实体类并映射两个不同的一对多关系,来表示多对多关系。

  1. class MyContext : DbContext
  2. {
  3. public DbSet<Post> Posts { get; set; }
  4. public DbSet<Tag> Tags { get; set; }
  5. protected override void OnModelCreating(ModelBuilder modelBuilder)
  6. {
  7. modelBuilder.Entity<PostTag>()
  8. .HasKey(t => new { t.PostId, t.TagId });
  9. modelBuilder.Entity<PostTag>()
  10. .HasOne(pt => pt.Post)
  11. .WithMany(p => p.PostTags)
  12. .HasForeignKey(pt => pt.PostId);
  13. modelBuilder.Entity<PostTag>()
  14. .HasOne(pt => pt.Tag)
  15. .WithMany(t => t.PostTags)
  16. .HasForeignKey(pt => pt.TagId);
  17. }
  18. }
  19. public class Post
  20. {
  21. public int PostId { get; set; }
  22. public string Title { get; set; }
  23. public string Content { get; set; }
  24. public List<PostTag> PostTags { get; set; }
  25. }
  26. public class Tag
  27. {
  28. public string TagId { get; set; }
  29. public List<PostTag> PostTags { get; set; }
  30. }
  31. public class PostTag
  32. {
  33. public int PostId { get; set; }
  34. public Post Post { get; set; }
  35. public string TagId { get; set; }
  36. public Tag Tag { get; set; }
  37. }