索引是跨多个数据存储区的常见概念。 尽管它们在数据存储中的实现可能会有所不同,但也可用于基于列(或一组列)更高效地进行查找。
不能使用数据批注创建索引。 您可以使用 "熟知 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
值。
protected override void OnModelCreating(ModelBuilder modelBuilder)
{
modelBuilder.Entity<Blog>()
.HasIndex(b => b.Url)
.IsUnique()
.HasFilter(null);
}
包含列
某些关系数据库允许配置一组列,这些列包含在索引中,但不是其 "键" 的一部分。 当查询中的所有列都作为键列或非键列包含在索引中时,这可以显著提高查询性能,因为表本身无需访问。 有关 SQL Server 包含列的详细信息,请参阅文档。
在下面的示例中,Url
列是索引键的一部分,因此对该列的任何查询筛选都可以使用索引。 但此外,仅访问 Title
和 PublishedOn
列的查询将不需要访问表,并且将更有效地运行:
protected override void OnModelCreating(ModelBuilder modelBuilder)
{
modelBuilder.Entity<Post>()
.HasIndex(p => p.Url)
.IncludeProperties(p => new
{
p.Title,
p.PublishedOn
});
}