EF Core索引

2020-11-10 14:31:58 浏览数 (1)

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

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

代码语言:javascript复制
protected override void OnModelCreating(ModelBuilder modelBuilder)
{
    modelBuilder.Entity<Blog>()
        .HasIndex(b => b.Url);
}

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

代码语言:javascript复制
protected override void OnModelCreating(ModelBuilder modelBuilder)
{
    modelBuilder.Entity<Person>()
        .HasIndex(p => new { p.FirstName, p.LastName });
}

 备注

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

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

索引唯一性

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

代码语言:javascript复制
protected override void OnModelCreating(ModelBuilder modelBuilder)
{
    modelBuilder.Entity<Blog>()
        .HasIndex(b => b.Url)
        .IsUnique();
}

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

索引名称

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

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

代码语言:javascript复制
protected override void OnModelCreating(ModelBuilder modelBuilder)
{
    modelBuilder.Entity<Blog>()
        .HasIndex(b => b.Url)
        .HasName("Index_Url");
}

索引筛选器

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

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

代码语言:javascript复制
protected override void OnModelCreating(ModelBuilder modelBuilder)
{
    modelBuilder.Entity<Blog>()
        .HasIndex(b => b.Url)
        .HasFilter("[Url] IS NOT NULL");
}

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

代码语言:javascript复制
protected override void OnModelCreating(ModelBuilder modelBuilder)
{
    modelBuilder.Entity<Blog>()
        .HasIndex(b => b.Url)
        .IsUnique()
        .HasFilter(null);
}

包含列

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

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

代码语言:javascript复制
protected override void OnModelCreating(ModelBuilder modelBuilder)
{
    modelBuilder.Entity<Post>()
        .HasIndex(p => p.Url)
        .IncludeProperties(p => new
        {
            p.Title,
            p.PublishedOn
        });
}

0 人点赞