乐趣区

关于spring:SpringBoot数据库管理-用Liquibase对数据库管理和迁移

Liquibase 是一个用于 用于跟踪、治理和利用数据库变动的开源工具 ,通过日志文件(changelog) 的模式记录数据库的变更 (changeset),而后执行日志文件中的批改,将数据库更新或回滚(rollback) 到统一的状态。它的指标是提供一种数据库类型无关的解决方案,通过执行 schema 类型的文件来达到迁徙。本文次要介绍 SpringBoot 与 Liquibase 的集成。@pdai

  • SpringBoot 数据库治理 – 用 Liquibase 对数据库治理和迁徙?

    • 常识筹备

      • 什么是 Liquibase?这类工具要解决什么问题?
      • Liquibase 有哪些概念?是如何工作的?
    • 简略示例

      • POM 依赖
      • yml 配置
      • 新增 changelog
      • 测试
    • 进一步了解

      • 比拟好的 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 lock
2022-04-12 20:41:20.737  INFO 8476 --- [main] liquibase.changelog                      : Creating database history table with name: test_db_liquibase.DATABASECHANGELOG
2022-04-12 20:41:20.783  INFO 8476 --- [main] liquibase.changelog                      : Reading from test_db_liquibase.DATABASECHANGELOG
Running Changeset: classpath:/db/changelog/db.changelog-master.yaml::20220412-01::pdai
2022-04-12 20:41:20.914  INFO 8476 --- [main] liquibase.changelog                      : Table person created
2022-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 53ms
Running Changeset: classpath:/db/changelog/db.changelog-master.yaml::20220412-02::pdai
2022-04-12 20:41:20.952  INFO 8476 --- [main] liquibase.changelog                      : Columns username(varchar(8)) added to person
2022-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 31ms
Running Changeset: classpath:/db/changelog/db.changelog-master.yaml::20220412-03::pdai
2022-04-12 20:41:21.351  INFO 8476 --- [main] liquibase.changelog                      : Lookup table added for person.state
2022-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 389ms
2022-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 lock
2022-04-12 20:49:01.761  INFO 9144 --- [main] liquibase.changelog                      : Reading from test_db_liquibase.DATABASECHANGELOG
2022-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 lock
2022-04-12 21:00:28.398  INFO 17540 --- [main] liquibase.changelog                      : Reading from test_db_liquibase.DATABASECHANGELOG
Running Changeset: classpath:/db/changelog/db.changelog-master.yaml::20220412-04::pdai
2022-04-12 21:00:28.516  INFO 17540 --- [main] liquibase.changelog                      : SQL in file classpath:/db/changelog/db.changelog-20220412-04.sql executed
2022-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 83ms
2022-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,最初部署。

示例源码

https://github.com/realpdai/t…

参考文章

https://docs.liquibase.com

退出移动版