写点什么

SpringBoot 数据库管理 - 用 Liquibase 对数据库管理和迁移?

作者:Java快了!
  • 2022 年 9 月 17 日
    湖南
  • 本文字数:5013 字

    阅读完需:约 16 分钟

Liquibase 是一个用于 用于跟踪、管理和应用数据库变化的开源工具 ,通过日志文件(changelog)的形式记录数据库的变更(changeset),然后执行日志文件中的修改,将数据库更新或回滚(rollback)到一致的状态。它的目标是提供一种数据库类型无关的解决方案,通过执行 schema 类型的文件来达到迁移。本文主要介绍 SpringBoot 与 Liquibase 的集成。@pdai

  • SpringBoot 数据库管理 - 用 Liquibase 对数据库管理和迁移?什么是 Liquibase?这类工具要解决什么问题?Liquibase 有哪些概念?是如何工作的?比较好的 changelog 的实践?除了 addColumn,addTable 还有哪些 changeType 呢?

知识准备

需要理解什么是 Liquibase,它的出现是要解决什么问题。

什么是 Liquibase?这类工具要解决什么问题?

Liquibase 是一个用于 用于跟踪、管理和应用数据库变化的开源工具 ,通过日志文件(changelog)的形式记录数据库的变更(changeset),然后执行日志文件中的修改,将数据库更新或回滚(rollback)到一致的状态。它的目标是提供一种数据库类型无关的解决方案,通过执行 schema 类型的文件来达到迁移。

其优点主要有以下:

  • 支持几乎所有主流的数据库,目前支持包括 Oracle/Sql Server/DB2/MySql/Sybase/PostgreSQL 等 各种数据库 ,这样在数据库的部署和升级环节可帮助应用系统支持更多数据库;

  • 支持版本控制,这样就能支持多开发者的协作维护;

  • 日志文件支持多种格式,如 XML, YAML, JSON, SQL 等;

  • 提供变化应用的回滚功能,可按时间、数量或标签(tag)回滚已应用的变化。通过这种方式,开发人员可轻易地还原数据库在任何时间点的状态

  • 支持多种运行方式,如命令行、Spring 集成、Maven 插件、Gradle 插件等。

为何会出现 Liquibase 这类工具呢?

在实际上线的应用中,随着版本的迭代,经常会遇到需要变更数据库表和字段,必然会遇到需要对这些变更进行记录和管理,以及回滚等等;同时只有脚本化且版本可管理,才能在让数据库实现真正的 DevOps(自动化执行 + 回滚等)。在这样的场景下 Liquibase 等工具的出现也就成为了必然。

Liquibase 有哪些概念?是如何工作的?

工作流程:将 SQL 变更记录到 changeset ,多个 changeset 变更组成了日志文件( changelog ),liquibase 将 changelog 更新日志文件同步到指定的 RDBMS 中。


日志文件(databaseChangeLog)支持多种格式,如 XML, YAML, JSON, SQL; 我们以 xml 为例,看下相关配置

<?xml version="1.0" encoding="UTF-8"?> <databaseChangeLog	xmlns="http://www.liquibase.org/xml/ns/dbchangelog"	xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"	xmlns:ext="http://www.liquibase.org/xml/ns/dbchangelog-ext"	xmlns:pro="http://www.liquibase.org/xml/ns/pro"	xsi:schemaLocation="http://www.liquibase.org/xml/ns/dbchangelog		http://www.liquibase.org/xml/ns/dbchangelog/dbchangelog-4.9.0.xsd		http://www.liquibase.org/xml/ns/dbchangelog-ext http://www.liquibase.org/xml/ns/dbchangelog/dbchangelog-ext.xsd		http://www.liquibase.org/xml/ns/pro http://www.liquibase.org/xml/ns/pro/liquibase-pro-4.9.0.xsd">    <changeSet id="1" author="bob">          <comment>A sample change log</comment>          <createTable/>     </changeSet>      <changeSet id="2" author="bob" runAlways="true">          <alterTable/>      </changeSet>      <changeSet id="3" author="alice" failOnError="false" dbms="oracle">        <alterTable/>      </changeSet>      <changeSet id="4" author="alice" failOnError="false" dbms="!oracle">        <alterTable/>      </changeSet>  </databaseChangeLog>
复制代码

简单示例

这里主要介绍基于 SpringBoot 集成 liquibase 来管理数据库的变更。

POM 依赖

Maven 包的依赖,主要包含 mysql 驱动, JDBC(这里 spring-boot-starter-data-jpa 包含了 jdbc 包,当然直接引入 jdbc 包也行),以及 liquibase 包。

<dependency>    <groupId>mysql</groupId>    <artifactId>mysql-connector-java</artifactId>    <version>8.0.28</version></dependency><dependency>    <groupId>com.github.wenhao</groupId>    <artifactId>jpa-spec</artifactId>    <version>3.1.0</version></dependency><dependency>    <groupId>org.springframework.boot</groupId>    <artifactId>spring-boot-starter-data-jpa</artifactId></dependency>
<dependency> <groupId>org.liquibase</groupId> <artifactId>liquibase-core</artifactId> <version>4.9.1</version></dependency>
复制代码

yml 配置

SpringBoot AutoConfig 默认已经包含了对 liquibase 的配置,在 spring.liquibase 配置下。

基础的配置,可以直接使用如下(主要是指定 change-log 的位置,默认的位置是 classpath:/db/changelog/db.changelog-master.yaml):

spring:  datasource:    url: jdbc:mysql://localhost:3306/test_db_liquibase?useSSL=false&autoReconnect=true&characterEncoding=utf8    driver-class-name: com.mysql.cj.jdbc.Driver    username: root    password: bfXa4Pt2lUUScy8jakXf  liquibase:    enabled: true    # 如下配置是被spring.datasource赋值的,所以可以不配置#    url: jdbc:mysql://localhost:3306/test_db_liquibase?useSSL=false&autoReconnect=true&characterEncoding=utf8#    user: root#    password: bfXa4Pt2lUUScy8jakXf    change-log: classpath:/db/changelog/db.changelog-master.yaml
复制代码

在开发时,更多的配置可以从如下 SpringBoot AutoConfig 中找到。


新增 changelog

XML 方式固然 OK,不过依然推荐使用 yml 格式。

databaseChangeLog:  - changeSet:      id: 20220412-01      author: pdai      changes:        - createTable:            tableName: person            columns:              - column:                  name: id                  type: int                  autoIncrement: true                  constraints:                    primaryKey: true                    nullable: false              - column:                  name: firstname                  type: varchar(50)              - column:                  name: lastname                  type: varchar(50)                  constraints:                    nullable: false              - column:                  name: state                  type: char(2)
- changeSet: id: 20220412-02 author: pdai changes: - addColumn: tableName: person columns: - column: name: username type: varchar(8)
- changeSet: id: 20220412-03 author: pdai changes: - addLookupTable: existingTableName: person existingColumnName: state newTableName: state newColumnName: id newColumnDataType: char(2)
复制代码

测试

启动 springBootApplication, 我们可以看到如下的几个 changeSet 被依次执行

2022-04-12 20:41:20.591  INFO 8476 --- [           main] liquibase.lockservice                    : Successfully acquired change log lock2022-04-12 20:41:20.737  INFO 8476 --- [           main] liquibase.changelog                      : Creating database history table with name: test_db_liquibase.DATABASECHANGELOG2022-04-12 20:41:20.783  INFO 8476 --- [           main] liquibase.changelog                      : Reading from test_db_liquibase.DATABASECHANGELOGRunning Changeset: classpath:/db/changelog/db.changelog-master.yaml::20220412-01::pdai2022-04-12 20:41:20.914  INFO 8476 --- [           main] liquibase.changelog                      : Table person created2022-04-12 20:41:20.914  INFO 8476 --- [           main] liquibase.changelog                      : ChangeSet classpath:/db/changelog/db.changelog-master.yaml::20220412-01::pdai ran successfully in 53msRunning Changeset: classpath:/db/changelog/db.changelog-master.yaml::20220412-02::pdai2022-04-12 20:41:20.952  INFO 8476 --- [           main] liquibase.changelog                      : Columns username(varchar(8)) added to person2022-04-12 20:41:20.952  INFO 8476 --- [           main] liquibase.changelog                      : ChangeSet classpath:/db/changelog/db.changelog-master.yaml::20220412-02::pdai ran successfully in 31msRunning Changeset: classpath:/db/changelog/db.changelog-master.yaml::20220412-03::pdai2022-04-12 20:41:21.351  INFO 8476 --- [           main] liquibase.changelog                      : Lookup table added for person.state2022-04-12 20:41:21.351  INFO 8476 --- [           main] liquibase.changelog                      : ChangeSet classpath:/db/changelog/db.changelog-master.yaml::20220412-03::pdai ran successfully in 389ms2022-04-12 20:41:21.382  INFO 8476 --- [           main] liquibase.lockservice                    : Successfully released change log lock
复制代码

查看数据库,你会发现数据已经变更


那我们如果重新启动这个 SpringBootApplication,会怎么呢?

很显然,因为 databasechangelog 表中已经有相关执行记录了,所以将不再执行变更

2022-04-12 20:49:01.566  INFO 9144 --- [           main] liquibase.lockservice                    : Successfully acquired change log lock2022-04-12 20:49:01.761  INFO 9144 --- [           main] liquibase.changelog                      : Reading from test_db_liquibase.DATABASECHANGELOG2022-04-12 20:49:01.812  INFO 9144 --- [           main] liquibase.lockservice                    : Successfully released change log lock
复制代码

进一步理解

通过这几个问题,进一步理解。

比较好的 changelog 的实践?

简单而言:yml 格式 + sql-file 方式

执行 sqlFile 格式的 changeSet,如下


执行的日志如下

2022-04-12 21:00:28.198  INFO 17540 --- [           main] liquibase.lockservice                    : Successfully acquired change log lock2022-04-12 21:00:28.398  INFO 17540 --- [           main] liquibase.changelog                      : Reading from test_db_liquibase.DATABASECHANGELOGRunning Changeset: classpath:/db/changelog/db.changelog-master.yaml::20220412-04::pdai2022-04-12 21:00:28.516  INFO 17540 --- [           main] liquibase.changelog                      : SQL in file classpath:/db/changelog/db.changelog-20220412-04.sql executed2022-04-12 21:00:28.516  INFO 17540 --- [           main] liquibase.changelog                      : ChangeSet classpath:/db/changelog/db.changelog-master.yaml::20220412-04::pdai ran successfully in 83ms2022-04-12 21:00:28.532  INFO 17540 --- [           main] liquibase.lockservice                    : Successfully released change log lock
复制代码

执行后,查看变更记录


数据表 user 表已经创建并插入一条数据


除了 addColumn,addTable 还有哪些 changeType 呢?

除了 addColumn,addTable 还有哪些 changeType 呢?

与此同时,还支持 如下 changeType :


此外,还支持执行 command

changeSet:    id:  executeCommand-example    author:  liquibase-docs    changes:    -  executeCommand:        args:        -  arg:            value:  -out        -  arg:            value:  -param2        executable:  mysqldump        os:  Windows 7        timeout:  10s
复制代码

比如,回滚的操作可以通过如下 command 进行


再比如,我们可以通过 Liquibase 来生成相关差异,再制作成 changeSet,最后部署。

用户头像

Java快了!

关注

还未添加个人签名 2022.09.03 加入

还未添加个人简介

评论

发布
暂无评论
SpringBoot数据库管理 - 用Liquibase对数据库管理和迁移?_数据库_Java快了!_InfoQ写作社区