基于模型的约定Model-Based Conventions

备注

仅限 EF6 及更高版本 - 此页面中讨论的功能、API 等已引入实体框架 6。 如果使用的是早期版本,则部分或全部信息不适用。

基于模型的约定是基于约定的模型配置的高级方法。 大多数情况下,应使用DbModelBuilder 上的自定义 Code First 约定 API 。 使用基于模型的约定之前,建议了解约定的 DbModelBuilder API。

基于模型的约定允许创建影响不能通过标准约定进行配置的属性和表的约定。 例如,表中的鉴别器列按层次结构模型和独立关联列。

创建约定Creating a Convention

创建基于模型的约定的第一步是在管道中选择约定需要应用到模型的时间。 有两种类型的模型约定:概念(C-空间)和存储区(S 空间)。 C-Space 约定应用于应用程序生成的模型,而 S 空间约定应用于表示数据库的模型的版本,并控制自动生成的列的命名方式。

模型约定是从 IConceptualModelConvention 或 IStoreModelConvention 扩展的类。 这些接口都接受类型为 MetadataItem 的泛型类型,该类型用于筛选约定适用的数据类型。

添加约定Adding a Convention

添加模型约定的方式与常规约定类相同。 在OnModelCreating方法中,将约定添加到模型的约定列表。

  1. using System.Data.Entity;
  2. using System.Data.Entity.Core.Metadata.Edm;
  3. using System.Data.Entity.Infrastructure;
  4. using System.Data.Entity.ModelConfiguration.Conventions;
  5. public class BlogContext : DbContext
  6. {
  7. public DbSet<Post> Posts { get; set; }
  8. public DbSet<Comment> Comments { get; set; }
  9. protected override void OnModelCreating(DbModelBuilder modelBuilder)
  10. {
  11. modelBuilder.Conventions.Add<MyModelBasedConvention>();
  12. }
  13. }

还可以使用 AddBefore<> 或 AddAfter<> 方法,与另一种约定相关添加约定。 有关实体框架应用的约定的详细信息,请参阅 “备注” 部分。

  1. protected override void OnModelCreating(DbModelBuilder modelBuilder)
  2. {
  3. modelBuilder.Conventions.AddAfter<IdKeyDiscoveryConvention>(new MyModelBasedConvention());
  4. }

示例:鉴别器模型约定Example: Discriminator Model Convention

重命名由 EF 生成的列是一个无法使用其他约定 Api 进行操作的示例。 这种情况下,使用模型约定是唯一的选择。

有关如何使用基于模型的约定配置生成的列的示例,请自定义鉴别器列的命名方式。 下面是基于模型的简单约定的一个示例,它将名为 “鉴别器” 的模型中的每一列都重命名为 “EntityType”。 这包括开发人员简单命名为 “鉴别器” 的列。 由于 “鉴别器” 列是生成的列,因此需要在 S 空间中运行。

  1. using System.Data.Entity;
  2. using System.Data.Entity.Core.Metadata.Edm;
  3. using System.Data.Entity.Infrastructure;
  4. using System.Data.Entity.ModelConfiguration.Conventions;
  5. class DiscriminatorRenamingConvention : IStoreModelConvention<EdmProperty>
  6. {
  7. public void Apply(EdmProperty property, DbModel model)
  8. {
  9. if (property.Name == "Discriminator")
  10. {
  11. property.Name = "EntityType";
  12. }
  13. }
  14. }

示例:常规 IA 重命名约定Example: General IA Renaming Convention

其他更复杂的基于模型的约定示例是配置独立关联(IAs)的命名方式。 这种情况下,模型约定适用,因为 IAs 由 EF 生成,并且不存在于 DbModelBuilder API 可以访问的模型中。

当 EF 生成 IA 时,它将创建一个名为 EntityType_KeyName 的列。 例如,对于名为 Customer 且名为 CustomerId 的键列的关联,它会生成一个名为 Customer_CustomerId 的列。 下面的约定从为 IA 生成的列名称中去除 “_“ 字符。

  1. using System.Data.Entity;
  2. using System.Data.Entity.Core.Metadata.Edm;
  3. using System.Data.Entity.Infrastructure;
  4. using System.Data.Entity.ModelConfiguration.Conventions;
  5. // Provides a convention for fixing the independent association (IA) foreign key column names.
  6. public class ForeignKeyNamingConvention : IStoreModelConvention<AssociationType>
  7. {
  8. public void Apply(AssociationType association, DbModel model)
  9. {
  10. // Identify ForeignKey properties (including IAs)
  11. if (association.IsForeignKey)
  12. {
  13. // rename FK columns
  14. var constraint = association.Constraint;
  15. if (DoPropertiesHaveDefaultNames(constraint.FromProperties, constraint.ToRole.Name, constraint.ToProperties))
  16. {
  17. NormalizeForeignKeyProperties(constraint.FromProperties);
  18. }
  19. if (DoPropertiesHaveDefaultNames(constraint.ToProperties, constraint.FromRole.Name, constraint.FromProperties))
  20. {
  21. NormalizeForeignKeyProperties(constraint.ToProperties);
  22. }
  23. }
  24. }
  25. private bool DoPropertiesHaveDefaultNames(ReadOnlyMetadataCollection<EdmProperty> properties, string roleName, ReadOnlyMetadataCollection<EdmProperty> otherEndProperties)
  26. {
  27. if (properties.Count != otherEndProperties.Count)
  28. {
  29. return false;
  30. }
  31. for (int i = 0; i < properties.Count; ++i)
  32. {
  33. if (!properties[i].Name.EndsWith("_" + otherEndProperties[i].Name))
  34. {
  35. return false;
  36. }
  37. }
  38. return true;
  39. }
  40. private void NormalizeForeignKeyProperties(ReadOnlyMetadataCollection<EdmProperty> properties)
  41. {
  42. for (int i = 0; i < properties.Count; ++i)
  43. {
  44. int underscoreIndex = properties[i].Name.IndexOf('_');
  45. if (underscoreIndex > 0)
  46. {
  47. properties[i].Name = properties[i].Name.Remove(underscoreIndex, 1);
  48. }
  49. }
  50. }
  51. }

扩展现有约定Extending Existing Conventions

如果需要编写一个约定,此约定与实体框架已应用于你的模型的约定之一类似,你始终可以扩展该约定,以避免必须从头开始重新编写该约定。 例如,将现有的 Id 匹配约定替换为自定义约定。 提高密钥约定的优点是:只有在未检测到或未显式配置任何键时才会调用重写的方法。 实体框架使用的约定列表如下所示: http://msdn.microsoft.com/library/system.data.entity.modelconfiguration.conventions.aspx

  1. using System.Data.Entity;
  2. using System.Data.Entity.Core.Metadata.Edm;
  3. using System.Data.Entity.Infrastructure;
  4. using System.Data.Entity.ModelConfiguration.Conventions;
  5. using System.Linq;
  6. // Convention to detect primary key properties.
  7. // Recognized naming patterns in order of precedence are:
  8. // 1. 'Key'
  9. // 2. [type name]Key
  10. // Primary key detection is case insensitive.
  11. public class CustomKeyDiscoveryConvention : KeyDiscoveryConvention
  12. {
  13. private const string Id = "Key";
  14. protected override IEnumerable<EdmProperty> MatchKeyProperty(
  15. EntityType entityType, IEnumerable<EdmProperty> primitiveProperties)
  16. {
  17. Debug.Assert(entityType != null);
  18. Debug.Assert(primitiveProperties != null);
  19. var matches = primitiveProperties
  20. .Where(p => Id.Equals(p.Name, StringComparison.OrdinalIgnoreCase));
  21. if (!matches.Any())
  22. {
  23. matches = primitiveProperties
  24. .Where(p => (entityType.Name + Id).Equals(p.Name, StringComparison.OrdinalIgnoreCase));
  25. }
  26. // If the number of matches is more than one, then multiple properties matched differing only by
  27. // case--for example, "Key" and "key".
  28. if (matches.Count() > 1)
  29. {
  30. throw new InvalidOperationException("Multiple properties match the key convention");
  31. }
  32. return matches;
  33. }
  34. }

接下来,我们需要在现有的密钥约定之前添加新约定。 添加 CustomKeyDiscoveryConvention 后,我们可以删除 IdKeyDiscoveryConvention。 如果我们未删除现有的 IdKeyDiscoveryConvention,则此约定仍优先于 Id 发现约定,因为它首先运行,但在未找到 “密钥” 属性的情况下,将运行 “Id” 约定。 我们会看到此行为,因为每个约定都将模型视为由上一个约定更新的(而不是单独对其进行操作并将所有组合在一起),因此,如果上一个约定更新了列名以匹配你的自定义约定的兴趣(早于该名称不重要),则它将应用于该列。

  1. public class BlogContext : DbContext
  2. {
  3. public DbSet<Post> Posts { get; set; }
  4. public DbSet<Comment> Comments { get; set; }
  5. protected override void OnModelCreating(DbModelBuilder modelBuilder)
  6. {
  7. modelBuilder.Conventions.AddBefore<IdKeyDiscoveryConvention>(new CustomKeyDiscoveryConvention());
  8. modelBuilder.Conventions.Remove<IdKeyDiscoveryConvention>();
  9. }
  10. }

注意Notes

此处的 MSDN 文档提供了实体框架当前应用的约定列表: http://msdn.microsoft.com/library/system.data.entity.modelconfiguration.conventions.aspx。 此列表是直接从源代码中提取的。 GitHub上提供了实体框架6的源代码,而实体框架所使用的许多约定都是基于自定义模型的约定的良好起点。