SpringBoot使用flyway初始化數(shù)據(jù)庫
Flyway這款數(shù)據(jù)庫版本工具就算大家沒有使用過但也略有耳聞了,SpringBoot對該款工具進(jìn)行集成的框架可以讓我們在啟動SpringBoot應(yīng)用時自動去找SQL版本文件進(jìn)行比對執(zhí)行,但在遷移或初始化時往往還是需要先手動進(jìn)行下數(shù)據(jù)庫的初始化配置,否則會把Unknown database的異常。 為了減少這一步所以個人就以SpringBoot的方式編碼在項目的啟動時自動進(jìn)行數(shù)據(jù)庫的初始化,然后再執(zhí)行版本文件。
自動建庫實現(xiàn)步驟具體思路 SpringBoot的配置項都會有相應(yīng)的Properties屬性類,數(shù)據(jù)庫的屬性類為DataSourceProperties,flyway的屬性類為FlywayProperties,盡可能的使用其中的配置項而不額外添加自定義的配置 建庫與相關(guān)設(shè)置的語句一般不會對已存在的設(shè)置進(jìn)行更改(如建庫建表時都是CREATE xxx IF NOT EXISTS),所以項目初始化時每次都執(zhí)行也不會影響現(xiàn)有的數(shù)據(jù)庫配置,編碼時可以不用添加執(zhí)行的前置判斷 代碼實現(xiàn)配置文件
spring: profiles: include: database
application-database.yml
spring: datasource: url: jdbc:mysql://yourIp:3306/spring_boot_series?useUnicode=true&characterEncoding=UTF-8&zeroDateTimeBehavior=convertToNull&allowMultiQueries=true&serverTimezone=GMT username: yourUsername password: yourPassword driver-class-name: com.mysql.cj.jdbc.Driver # 自動讀取spring.datasource配置進(jìn)行遷移操作 flyway: # 版本遷移位置 locations: classpath:db baseline-version: 1.0.2 init-sqls: - SET @OLD_UNIQUE_CHECKS = @@UNIQUE_CHECKS, UNIQUE_CHECKS = 0; - SET @OLD_FOREIGN_KEY_CHECKS = @@FOREIGN_KEY_CHECKS, FOREIGN_KEY_CHECKS = 0; - SET @OLD_SQL_MODE = @@SQL_MODE, SQL_MODE = ’ONLY_FULL_GROUP_BY,STRICT_TRANS_TABLES,NO_ZERO_IN_DATE,NO_ZERO_DATE,ERROR_FOR_DIVISION_BY_ZERO,NO_ENGINE_SUBSTITUTION’; - CREATE SCHEMA IF NOT EXISTS `spring_boot_series` DEFAULT CHARACTER SET utf8 COLLATE utf8_general_ci;
為了讓項目的配置結(jié)構(gòu)更清晰,所以我把數(shù)據(jù)庫的相關(guān)配置都獨立到application-database.yml中。flyway的init-sqls配置是當(dāng)獲取到SQL statement時會執(zhí)行SQL語句列表,但需要注意的是如果spring.datasource.url連接不上,該塊語句是依舊無法被執(zhí)行的,因為SpringBoot集成的flyway配置是通過spring.datasource的配置去連接數(shù)據(jù)庫的,如果url中的數(shù)據(jù)庫不存在,flyway的版本遷移就無法執(zhí)行了。前文提到盡可能的使用SpringBoot原有的配置項,所以init-sqls其實是我配給自己用的。 個人通過init-sqls配置初始化語句與其它方式配置初始化想到的好處:
無需將初始化語句硬編碼再讓Statement執(zhí)行 當(dāng)數(shù)據(jù)庫選型進(jìn)行變更時只需再配置文件中進(jìn)行修改相應(yīng)的初始化語句 SpringBoot可以直接以字符串列表進(jìn)行讀取,開發(fā)過程中使用起來更加靈活,而不用去解析SQL文件 數(shù)據(jù)庫初始化器DatabaseInitializerpackage io.wilson.flyway;import com.zaxxer.hikari.HikariDataSource;import lombok.AllArgsConstructor;import lombok.extern.slf4j.Slf4j;import org.springframework.boot.autoconfigure.flyway.FlywayProperties;import org.springframework.boot.autoconfigure.jdbc.DataSourceProperties;import org.springframework.stereotype.Component;import javax.annotation.PostConstruct;import java.sql.Connection;import java.sql.SQLException;import java.sql.Statement;/** * @author Wilson */@Slf4j@Component@AllArgsConstructorpublic class DatabaseInitializer { private final FlywayProperties flywayProperties; private final DataSourceProperties dataSourceProperties; @PostConstruct public void init() throws SQLException { log.info('DatabaseInitializer uses flyway init-sqls to initiate database'); String url = dataSourceProperties.getUrl(); // jdbc url最后一個 ’/’ 用于分割具體 schema?參數(shù) int lastSplitIndex = url.lastIndexOf(’/’); // 獲取spring.datasource.url具體數(shù)據(jù)庫schema前的jdbc url String addressUrl = url.substring(0, lastSplitIndex); // 直連數(shù)據(jù)庫地址:jdbc:mysql://yourIp:port HikariDataSource dataSource = new HikariDataSource(); dataSource.setJdbcUrl(addressUrl); dataSource.setUsername(dataSourceProperties.getUsername()); dataSource.setPassword(dataSourceProperties.getPassword()); Connection connection = dataSource.getConnection(); Statement statement = connection.createStatement(); for (String sql : flywayProperties.getInitSqls()) { // 通過flyway的init-sqls配置進(jìn)行建庫與數(shù)據(jù)庫配置 // executeUpdate:執(zhí)行給定的SQL語句,該語句可以是INSERT,UPDATE或DELETE語句或不返回任何內(nèi)容的SQL語句,例如SQL DDL語句。 statement.executeUpdate(sql); } statement.close(); connection.close(); dataSource.close(); log.info('DatabaseInitializer initialize completed'); }}簡單的主程序FlywayApplication
package io.wilson.flyway;import org.mybatis.spring.annotation.MapperScan;import org.springframework.boot.SpringApplication;import org.springframework.boot.autoconfigure.SpringBootApplication;/** * @author Wilson */@SpringBootApplication@MapperScan('io.wilson.flyway.mapper')public class FlywayApplication { public static void main(String[] args) { SpringApplication.run(FlywayApplication.class, args); }}執(zhí)行效果圖
當(dāng)項目啟動時可以看到DatabaseInitializer會先通過init-sqls進(jìn)行數(shù)據(jù)庫的初始化,然后SpringBoot再初始化通用的數(shù)據(jù)源并執(zhí)行SQL版本文件。圖中的create_database.sql文件存放的是init-sqls配置的語句,該文件不會被執(zhí)行,可以忽略。
以上就是SpringBoot使用flyway初始化數(shù)據(jù)庫的詳細(xì)內(nèi)容,更多關(guān)于SpringBoot 初始化數(shù)據(jù)庫的資料請關(guān)注好吧啦網(wǎng)其它相關(guān)文章!
相關(guān)文章:
1. python如何換行輸出2. Python使用urlretrieve實現(xiàn)直接遠(yuǎn)程下載圖片的示例代碼3. Python:UserWarning:此模式具有匹配組。要實際獲得組,請使用str.extract4. Android Studio中一套代碼多渠道打包的實現(xiàn)方法5. Java 接口和抽象類的區(qū)別詳解6. python如何計算圓的面積7. Java使用Tesseract-Ocr識別數(shù)字8. Android打包篇:Android Studio將代碼打包成jar包教程9. 詳解java google Thumbnails 圖片處理10. 解決Android Studio 格式化 Format代碼快捷鍵問題
