数据库初始化
SQL 数据库的初始化方式取决于你的技术栈。 当然,如果数据库是独立进程,也可以手动初始化。 推荐使用单一机制进行模式(schema)生成。
使用 Hibernate 初始化数据库
你可以通过设置 spring.jpa.hibernate.ddl-auto 控制 Hibernate 的数据库初始化。
支持的取值有 none、validate、update、create 和 create-drop。
Spring Boot 会根据你是否使用嵌入式数据库为你选择默认值。
嵌入式数据库通过 Connection 类型和 JDBC url 识别。
hsqldb、h2 或 derby 属于嵌入式数据库,其他则不是。
如果检测到嵌入式数据库且未检测到 schema 管理器(Flyway 或 Liquibase),ddl-auto 默认值为 create-drop。
其他情况下,默认为 none。
从内存数据库切换到“真实”数据库时,请注意不要假设新平台上表和数据一定存在。
你要么显式设置 ddl-auto,要么使用其他机制初始化数据库。
你可以通过启用 org.hibernate.SQL logger 输出 schema 创建语句。
如果你启用了 debug 模式,系统会自动为你完成此操作。
|
此外,如果 Hibernate 从零创建 schema(即 ddl-auto 属性为 create 或 create-drop),classpath 根目录下名为 import.sql 的文件会在启动时被执行。
这对于演示和测试很有用(需谨慎),但通常不建议在生产环境的 classpath 中存在。
这是 Hibernate 的特性(与 Spring 无关)。
使用基础 SQL 脚本初始化数据库
Spring Boot 可以自动为你的 JDBC DataSource 或 R2DBC ConnectionFactory 创建 schema(DDL 脚本)并初始化数据(DML 脚本)。
默认情况下,它会从 optional:classpath*:schema.sql 加载 schema 脚本,从 optional:classpath*:data.sql 加载数据脚本。
这些 schema 和数据脚本的位置可分别通过 spring.sql.init.schema-locations 和 spring.sql.init.data-locations 自定义。
optional: 前缀表示即使文件不存在,应用也会启动。
如果希望文件缺失时应用启动失败,移除 optional: 前缀即可。
此外,Spring Boot 还会处理 optional:classpath*:schema-${platform}.sql 和 optional:classpath*:data-${platform}.sql 文件(如存在),其中 ${platform} 取自 spring.sql.init.platform。
这样可以根据需要切换为数据库专用脚本。
例如,你可以将其设置为数据库厂商名(如 hsqldb、h2、oracle、mysql、postgresql 等)。
默认情况下,只有使用嵌入式内存数据库时才会执行 SQL 数据库初始化。
如需无论数据库类型都初始化,设置 spring.sql.init.mode 为 always。
如需禁用初始化,设置 spring.sql.init.mode 为 never。
默认情况下,Spring Boot 启用了基于脚本的数据库初始化器的 fail-fast 特性。
即如果脚本执行异常,应用启动会失败。
你可以通过设置 spring.sql.init.continue-on-error 调整该行为。
基于脚本的 DataSource 初始化默认在任何 JPA EntityManagerFactory bean 创建之前执行。
schema.sql 可用于为 JPA 管理的实体创建 schema,data.sql 可用于填充数据。
虽然我们不推荐同时使用多种数据源初始化技术,但如果你希望基于脚本的 DataSource 初始化能在 Hibernate 创建 schema 后继续构建,可将 spring.jpa.defer-datasource-initialization 设为 true。
这样会延迟数据源初始化,直到所有 EntityManagerFactory bean 创建并初始化完成。
此时 schema.sql 可用于对 Hibernate 创建的 schema 进行补充,data.sql 可用于填充数据。
初始化脚本支持 -- 单行注释和 /* */ 块注释。
不支持其他注释格式。
|
如果你使用 高级数据库迁移工具(如 Flyway 或 Liquibase),应仅用它们来创建和初始化 schema。
不推荐将基础的 schema.sql 和 data.sql 脚本与 Flyway 或 Liquibase 混用,且未来版本将移除该支持。
初始化 Spring Batch 数据库
如果你使用 Spring Batch,框架已为大多数主流数据库平台预置了 SQL 初始化脚本。 Spring Boot 能自动检测你的数据库类型并在启动时执行这些脚本。 如果使用嵌入式数据库,默认会自动初始化。 你也可以为任意数据库类型启用,如下例所示:
-
Properties
-
YAML
spring.batch.jdbc.initialize-schema=always
spring:
batch:
jdbc:
initialize-schema: "always"
你也可以通过将 spring.batch.jdbc.initialize-schema 显式设置为 never 来关闭初始化。
使用高级数据库迁移工具
启动时执行 Flyway 数据库迁移
如需在启动时自动运行 Flyway 数据库迁移,将相应的 Flyway 模块加入 classpath。
内存和文件型数据库由 org.flywaydb:flyway-core 支持。
其他数据库则需对应的数据库专用模块。
例如,PostgreSQL 用 org.flywaydb:flyway-database-postgresql,MySQL 用 org.flywaydb:flyway-mysql。
详见 Flyway 文档。
通常,迁移脚本格式为 V<VERSION>__<NAME>.sql(<VERSION> 为下划线分隔的版本号,如 1 或 2_1)。
默认目录为 classpath:db/migration,可通过设置 spring.flyway.locations 修改。
该属性为逗号分隔的一个或多个 classpath: 或 filesystem: 路径。
例如,以下配置会在默认 classpath 目录和 /opt/migration 目录查找脚本:
-
Properties
-
YAML
spring.flyway.locations=classpath:db/migration,filesystem:/opt/migration
spring:
flyway:
locations: "classpath:db/migration,filesystem:/opt/migration"
你还可以添加特殊的 {vendor} 占位符以使用数据库专用脚本。
假设如下配置:
-
Properties
-
YAML
spring.flyway.locations=classpath:db/migration/{vendor}
spring:
flyway:
locations: "classpath:db/migration/{vendor}"
上述配置会根据数据库类型(如 MySQL 为 db/migration/mysql)选择目录。
支持的数据库列表见 DatabaseDriver。
迁移也可用 Java 编写。
Flyway 会自动配置实现了 JavaMigration 的 bean。
FlywayProperties 提供了大部分 Flyway 设置及少量额外属性,可用于禁用迁移或关闭位置检查。
如需更细致的配置,可注册 FlywayConfigurationCustomizer bean。
Spring Boot 通过调用 Flyway.migrate() 执行数据库迁移。
如需更大控制权,可提供实现 FlywayMigrationStrategy 的 @Bean。
Flyway 支持 SQL 和 Java 回调。
如用 SQL 回调,将脚本放在 classpath:db/migration 目录。
如用 Java 回调,实现 Callback 的 bean 即可。
这些 bean 会自动注册到 Flyway。
可通过 @Order 注解或实现 Ordered 排序。
默认情况下,Flyway 会自动装配上下文中的(@Primary)DataSource 并用于迁移。
如需使用其他 DataSource,可自行创建并用 @Bean 标记为 @FlywayDataSource。
如需保留主自动配置的数据源并共存,记得将 @Bean 注解的 defaultCandidate 属性设为 false。
或者,也可通过在外部属性中设置 spring.flyway.[url,user,password] 使用 Flyway 的原生 DataSource。
只要设置了 spring.flyway.url 或 spring.flyway.user,Flyway 就会使用自己的 DataSource。
如三者有未设置,则会使用对应的 spring.datasource 属性值。
你还可以用 Flyway 针对特定场景提供数据。
例如,将测试专用迁移脚本放在 src/test/resources,仅在测试启动时运行。
也可用 profile 专属配置自定义 spring.flyway.locations,使某些迁移仅在特定 profile 激活时运行。
如在 application-dev.properties 中可指定:
-
Properties
-
YAML
spring.flyway.locations=classpath:/db/migration,classpath:/dev/db/migration
spring:
flyway:
locations: "classpath:/db/migration,classpath:/dev/db/migration"
这样,dev/db/migration 下的迁移只会在 dev profile 激活时运行。
启动时执行 Liquibase 数据库迁移
如需在启动时自动运行 Liquibase 数据库迁移,将 org.liquibase:liquibase-core 加入 classpath。
|
当你将 |
默认情况下,主变更日志读取自 db/changelog/db.changelog-master.yaml,可通过设置 spring.liquibase.change-log 修改。
除了 YAML,Liquibase 还支持 JSON、XML 和 SQL 变更日志格式。
默认情况下,Liquibase 会自动装配上下文中的(@Primary)DataSource 并用于迁移。
如需使用其他 DataSource,可自行创建并用 @Bean 标记为 @LiquibaseDataSource。
如需保留主自动配置的数据源并共存,记得将 @Bean 注解的 defaultCandidate 属性设为 false。
或者,也可通过在外部属性中设置 spring.liquibase.[driver-class-name,url,user,password] 使用 Liquibase 的原生 DataSource。
只要设置了 spring.liquibase.url 或 spring.liquibase.user,Liquibase 就会使用自己的 DataSource。
如三者有未设置,则会使用对应的 spring.datasource 属性值。
更多可用设置(如 contexts、默认 schema 等)详见 LiquibaseProperties。
如需在使用前自定义 Liquibase 实例,可使用 Customizer<Liquibase> bean。
仅用于测试的 Flyway 迁移
如需创建用于填充测试数据库的 Flyway 迁移,将其放在 src/test/resources/db/migration。
例如,src/test/resources/db/migration/V9999__test-data.sql 文件会在生产迁移后且仅在运行测试时执行。
你可以用该文件创建所需测试数据。
该文件不会被打包进 uber jar 或容器。
仅用于测试的 Liquibase 迁移
如需创建用于填充测试数据库的 Liquibase 迁移,你需要创建一个测试 changelog,并包含生产 changelog。
首先,需要配置 Liquibase 在运行测试时使用不同的 changelog。
一种方式是创建 Spring Boot 的 test profile,并将 Liquibase 属性放入其中。
为此,在 src/test/resources/application-test.properties 文件中添加如下属性:
-
Properties
-
YAML
spring.liquibase.change-log=classpath:/db/changelog/db.changelog-test.yaml
spring:
liquibase:
change-log: "classpath:/db/changelog/db.changelog-test.yaml"
这样配置后,Liquibase 在 test profile 下会使用不同的 changelog。
接下来,在 src/test/resources/db/changelog/db.changelog-test.yaml 路径下创建 changelog 文件:
databaseChangeLog:
- include:
file: classpath:/db/changelog/db.changelog-master.yaml
- changeSet:
runOrder: "last"
id: "test"
changes:
# 在此插入你的变更
该 changelog 会在测试运行时被使用,不会被打包进 uber jar 或容器。
它包含生产 changelog,并声明一个新的 changeset,其中 runOrder: last 表示该 changeset 会在所有生产 changeset 之后执行。
你可以使用 insert changeset 插入数据,或用 sql changeset 直接执行 SQL。
最后,需要配置 Spring Boot 在运行测试时激活 test profile。
可在带有 @SpringBootTest 注解的测试类上添加 @ActiveProfiles("test") 注解。
依赖已初始化的数据库
数据库初始化作为应用上下文刷新的一部分,在应用启动时执行。 为便于在启动期间访问已初始化的数据库,作为数据库初始化器的 bean 以及依赖数据库已初始化的 bean 会被自动检测。 依赖数据库初始化的 bean 会被配置为依赖于初始化数据库的 bean。 如果应用在启动期间尝试访问尚未初始化的数据库,可以配置额外的 bean 检测机制,以确保初始化顺序。
检测数据库初始化器
Spring Boot 会自动检测以下类型的 bean 作为 SQL 数据库初始化器:
如果你使用了第三方数据库初始化库的 starter,可能会自动检测其他类型的 bean。
如需检测其他 bean,可在 META-INF/spring.factories 中注册 DatabaseInitializerDetector 的实现。
检测依赖数据库初始化的 Bean
Spring Boot 会自动检测以下类型的 bean 作为依赖数据库初始化的 bean:
-
AbstractEntityManagerFactoryBean(除非spring.jpa.defer-datasource-initialization设为true) -
DSLContext(jOOQ) -
EntityManagerFactory(除非spring.jpa.defer-datasource-initialization设为true)
如果你使用了第三方数据访问库的 starter,可能会自动检测其他类型的 bean。
如需检测其他 bean,可在 META-INF/spring.factories 中注册 DependsOnDatabaseInitializationDetector 的实现。
或者,也可以在 bean 的类或其 @Bean 方法上添加 @DependsOnDatabaseInitialization 注解。