数据种子设定Data Seeding

数据种子是用初始数据集填充数据库的过程。

可以通过多种方式在 EF Core 中完成此操作:

  • 模型种子数据
  • 手动迁移自定义
  • 自定义初始化逻辑

模型种子数据Model seed data

备注

此功能是 EF Core 2.1 中的新增功能。

与在 EF6 中不同,在 EF Core 中,种子设定数据可以作为模型配置的一部分与实体类型相关联。 然后 EF Core迁移可以自动计算在将数据库升级到新版本的模型时需要应用的插入、更新或删除操作。

备注

迁移仅在确定应该执行哪种操作以使种子数据进入所需状态时才考虑模型更改。 因此,在迁移外部执行的数据更改可能会丢失或导致错误。

例如,这将为 OnModelCreating中的 Blog 配置种子数据:

  1. modelBuilder.Entity<Blog>().HasData(new Blog {BlogId = 1, Url = "http://sample.com"});

若要添加具有关系的实体,需要指定外键值:

  1. modelBuilder.Entity<Post>().HasData(
  2. new Post() { BlogId = 1, PostId = 1, Title = "First post", Content = "Test 1" });

如果实体类型具有隐藏状态的任何属性,则可以使用匿名类提供值:

  1. modelBuilder.Entity<Post>().HasData(
  2. new { BlogId = 1, PostId = 2, Title = "Second post", Content = "Test 2" });

拥有的实体类型可以采用类似的方式进行种子设定:

  1. modelBuilder.Entity<Post>().OwnsOne(p => p.AuthorName).HasData(
  2. new { PostId = 1, First = "Andriy", Last = "Svyryd" },
  3. new { PostId = 2, First = "Diego", Last = "Vega" });

有关更多上下文,请参阅完整的示例项目

将数据添加到模型后,应使用迁移来应用更改。

提示

如果需要在自动部署中应用迁移,可以创建一个可在执行前预览的SQL 脚本

或者,可以使用 context.Database.EnsureCreated() 创建包含种子数据的新数据库,例如测试数据库,或使用内存中提供程序或任何非关系数据库时。 请注意,如果数据库已存在,EnsureCreated() 将不会更新数据库中的架构或种子数据。 对于关系数据库,如果您计划使用迁移,则不应调用 EnsureCreated()

模型种子数据的限制Limitations of model seed data

此类型的种子数据由迁移管理,需要在不连接到数据库的情况下生成用于更新数据库中已存在的数据的脚本。 这会施加一些限制:

  • 主键值需要指定,即使它通常由数据库生成。 它用于检测迁移间的数据更改。
  • 如果以任何方式更改了主键,则将删除以前的种子数据。

因此,此功能最适用于不需要在迁移外更改的静态数据,并且不依赖于数据库中的任何其他内容(例如,邮政编码)。

如果你的方案包括以下任何一种情况,则建议使用上一部分中所述的自定义初始化逻辑:

  • 用于测试的临时数据
  • 依赖于数据库状态的数据
  • 需要由数据库生成的键值的数据,包括使用替代密钥作为标识的实体
  • 需要自定义转换的数据(不由值转换处理),如某些密码哈希
  • 需要调用外部 API 的数据,例如 ASP.NET Core 标识角色和用户创建

手动迁移自定义Manual migration customization

添加迁移时,对使用 HasData 指定的数据所做的更改将转换为对 InsertData()UpdateData()DeleteData()的调用。 解决 HasData 的某些限制的一种方法是手动将这些调用或自定义操作添加到迁移。

  1. migrationBuilder.InsertData(
  2. table: "Blogs",
  3. columns: new[] { "Url" },
  4. values: new object[] { "http://generated.com" });

自定义初始化逻辑Custom initialization logic

执行数据种子设定的一种简单而有效的方法是在主应用程序逻辑开始执行之前使用DbContext.SaveChanges()

  1. using (var context = new DataSeedingContext())
  2. {
  3. context.Database.EnsureCreated();
  4. var testBlog = context.Blogs.FirstOrDefault(b => b.Url == "http://test.com");
  5. if (testBlog == null)
  6. {
  7. context.Blogs.Add(new Blog { Url = "http://test.com" });
  8. }
  9. context.SaveChanges();
  10. }

警告

种子设定代码不应是正常应用执行的一部分,因为这可能会导致多个实例运行时出现并发性问题,并且还要求应用有权修改数据库架构。

根据部署的约束,可以通过不同的方式执行初始化代码:

  • 在本地运行初始化应用程序
  • 将初始化应用与主应用一起部署,调用初始化例程,禁用或删除初始化应用。

通常可以使用发布配置文件自动完成此配置。