在使用EF Core进行C#数据库开发时,索引是提升查询效率的核心手段,通过EF Core的Fluent API或者数据注解可以灵活配置索引,同时需要遵循合理的优化原则避免索引带来的性能损耗。
EF Core中索引的基础创建方式
使用Fluent API创建索引
EF Core的Fluent API是配置索引最常用的方式,支持单字段索引、复合索引等多种类型,在DbContext的OnModelCreating方法中配置即可。
创建单字段普通索引的示例代码如下:
using Microsoft.EntityFrameworkCore;
using System;
using System.Collections.Generic;
namespace EFCoreIndexDemo
{
// 定义实体类
public class User
{
public int Id { get; set; }
public string UserName { get; set; }
public string Email { get; set; }
public DateTime CreateTime { get; set; }
}
// 定义数据库上下文
public class AppDbContext : DbContext
{
public DbSet<User> Users { get; set; }
protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder)
{
// 配置数据库连接,这里使用SQL Server示例
optionsBuilder.UseSqlServer("Server=127.0.0.1;Database=TestDb;User Id=sa;Password=123456;");
}
protected override void OnModelCreating(ModelBuilder modelBuilder)
{
// 为UserName字段创建普通索引
modelBuilder.Entity<User>()
.HasIndex(u => u.UserName)
.HasDatabaseName("IX_User_UserName"); // 指定索引名称
base.OnModelCreating(modelBuilder);
}
}
}
创建唯一索引
如果需要保证字段值的唯一性,同时提升查询效率,可以创建唯一索引,只需要在索引配置中添加IsUnique方法即可。
protected override void OnModelCreating(ModelBuilder modelBuilder)
{
// 为Email字段创建唯一索引,保证邮箱不重复
modelBuilder.Entity<User>()
.HasIndex(u => u.Email)
.IsUnique() // 设置为唯一索引
.HasDatabaseName("IX_User_Email_Unique");
}
创建复合索引
当查询经常同时使用多个字段作为条件时,创建复合索引比多个单字段索引效率更高,Fluent API支持传入多个字段配置复合索引。
protected override void OnModelCreating(ModelBuilder modelBuilder)
{
// 创建UserName和CreateTime的复合索引,适合按用户名和时间范围查询的场景
modelBuilder.Entity<User>()
.HasIndex(u => new { u.UserName, u.CreateTime })
.HasDatabaseName("IX_User_UserName_CreateTime");
}
EF Core索引优化方法
避免过度创建索引
索引虽然能提升查询效率,但会增加数据插入、更新、删除的开销,同时占用额外的存储空间。需要根据实际查询频率决定索引创建,优先为高频查询的字段创建索引,低频查询字段不需要创建。
选择合适的索引列
优先为高区分度的字段创建索引,比如用户表的用户名字段,不同用户用户名重复率低,索引效果好。而性别这类只有少数取值的字段,创建索引的收益很低,不建议创建。
控制复合索引的字段顺序
复合索引的字段顺序会影响使用效果,最常用作查询条件的字段放在前面,范围查询的字段放在后面。比如经常按UserName等值查询,再按CreateTime范围查询,那么复合索引应该把UserName放在前面。
定期维护索引
随着数据的增删改,索引会产生碎片,定期重建或重组索引可以保持索引性能。可以通过EF Core执行原始SQL语句完成索引维护,示例代码如下:
using (var db = new AppDbContext())
{
// 重建索引的SQL语句,不同数据库语法有差异,这里是SQL Server示例
db.Database.ExecuteSqlRaw("ALTER INDEX IX_User_UserName ON Users REBUILD;");
}
索引创建注意事项
使用EF Core创建索引时,索引配置会在数据迁移时应用到数据库,所以修改索引配置后需要生成并执行新的迁移文件。另外,删除索引也需要通过Fluent API移除对应配置,然后生成迁移更新数据库。
如果需要对现有数据库添加索引而不想使用迁移,也可以通过EF Core执行原始SQL语句直接创建索引,这种方式适合已经上线的项目临时优化索引的场景。