索引Indexes

索引是跨多个数据存储区的常见概念。 尽管它们在数据存储中的实现可能会有所不同,但也可用于基于列(或一组列)更高效地进行查找。

不能使用数据批注创建索引。 您可以使用 “熟知 API” 按如下方式为单个列指定索引:

  1. protected override void OnModelCreating(ModelBuilder modelBuilder)
  2. {
  3. modelBuilder.Entity<Blog>()
  4. .HasIndex(b => b.Url);
  5. }

您还可以为多个列指定索引:

  1. protected override void OnModelCreating(ModelBuilder modelBuilder)
  2. {
  3. modelBuilder.Entity<Person>()
  4. .HasIndex(p => new { p.FirstName, p.LastName });
  5. }

备注

按照约定,将在用作外键的每个属性(或一组属性)中创建索引。

EF Core 每个不同的属性集仅支持一个索引。 如果使用 “熟知 API” 来配置已定义索引的属性集的索引(按照约定或以前的配置),则会更改该索引的定义。 如果要进一步配置由约定创建的索引,则此操作非常有用。

索引唯一性Index uniqueness

默认情况下,索引不唯一:允许多行具有与索引的列集相同的值。 可以使索引唯一,如下所示:

  1. protected override void OnModelCreating(ModelBuilder modelBuilder)
  2. {
  3. modelBuilder.Entity<Blog>()
  4. .HasIndex(b => b.Url)
  5. .IsUnique();
  6. }

尝试为索引的列集插入多个具有相同值的实体将导致引发异常。

索引名称Index name

按照约定,在关系数据库中创建的索引将命名为 IX_<type name>_<property name>。 对于复合索引,<property name> 变成以下划线分隔的属性名称列表。

您可以使用 “熟知 API” 设置在数据库中创建的索引的名称:

  1. protected override void OnModelCreating(ModelBuilder modelBuilder)
  2. {
  3. modelBuilder.Entity<Blog>()
  4. .HasIndex(b => b.Url)
  5. .HasName("Index_Url");
  6. }

索引筛选器Index filter

某些关系数据库允许您指定筛选索引或部分索引。 这使您可以只为列的值的一个子集编制索引,从而减少索引的大小并改善性能和磁盘空间的使用情况。 有关 SQL Server 筛选索引的详细信息,请参阅文档

您可以使用熟知的 API 来指定索引的筛选器,作为 SQL 表达式提供:

  1. protected override void OnModelCreating(ModelBuilder modelBuilder)
  2. {
  3. modelBuilder.Entity<Blog>()
  4. .HasIndex(b => b.Url)
  5. .HasFilter("[Url] IS NOT NULL");
  6. }

当使用 SQL Server 提供程序 EF 时,将为唯一索引中包含的所有可以为 null 的列添加 'IS NOT NULL' 筛选器。 若要重写此约定,可以提供 null 值。

  1. protected override void OnModelCreating(ModelBuilder modelBuilder)
  2. {
  3. modelBuilder.Entity<Blog>()
  4. .HasIndex(b => b.Url)
  5. .IsUnique()
  6. .HasFilter(null);
  7. }

包含列Included columns

某些关系数据库允许配置一组列,这些列包含在索引中,但不是其 “键” 的一部分。 当查询中的所有列都作为键列或非键列包含在索引中时,这可以显著提高查询性能,因为表本身无需访问。 有关 SQL Server 包含列的详细信息,请参阅文档

在下面的示例中,Url 列是索引键的一部分,因此对该列的任何查询筛选都可以使用索引。 但此外,仅访问 TitlePublishedOn 列的查询将不需要访问表,并且将更有效地运行:

  1. protected override void OnModelCreating(ModelBuilder modelBuilder)
  2. {
  3. modelBuilder.Entity<Post>()
  4. .HasIndex(p => p.Url)
  5. .IncludeProperties(p => new
  6. {
  7. p.Title,
  8. p.PublishedOn
  9. });
  10. }