国产av日韩一区二区三区精品,成人性爱视频在线观看,国产,欧美,日韩,一区,www.成色av久久成人,2222eeee成人天堂

首頁(yè) Java java教程 多模組 Gradle 專(zhuān)案中的 Flyway 遷移(乾淨(jìng)架構(gòu))

多模組 Gradle 專(zhuān)案中的 Flyway 遷移(乾淨(jìng)架構(gòu))

Jan 19, 2025 am 08:05 AM

使用 Flyway 在 Java 中自動(dòng)執(zhí)行資料庫(kù)遷移

資料庫(kù)遷移是軟體開(kāi)發(fā)的重要方面,特別是在持續(xù)整合和交付 (CI/CD) 為標(biāo)準(zhǔn)實(shí)踐的環(huán)境中。隨著應(yīng)用程式的成長(zhǎng)和發(fā)展,它所依賴(lài)的資料庫(kù)模式也必須如此。手動(dòng)管理這些架構(gòu)變??更可能會(huì)導(dǎo)致錯(cuò)誤並消耗大量時(shí)間。

Flyway 登場(chǎng),這是一款專(zhuān)為簡(jiǎn)化資料庫(kù)遷移而客製化的寶貴開(kāi)源工具。 Flyway 為您的資料庫(kù)引入了版本控制,使您能夠安全可靠地遷移架構(gòu)。在本文中,我們將探索如何使用 Flyway 在多模組 gragle java 專(zhuān)案中自動(dòng)化資料庫(kù)遷移,確保管理資料庫(kù)變更成為一個(gè)簡(jiǎn)化的、防錯(cuò)的過(guò)程。

更多有關(guān)飛行路線(xiàn)的詳細(xì)資訊

了解 Gradle 中的多項(xiàng)目構(gòu)建

雖然一些較小的專(zhuān)案或整體應(yīng)用程式可能僅使用一個(gè)建置檔案和統(tǒng)一的來(lái)源結(jié)構(gòu)進(jìn)行管理,但較大的專(zhuān)案經(jīng)常被組織成多個(gè)相互依賴(lài)的模組。術(shù)語(yǔ)「相互依賴(lài)」是這裡的關(guān)鍵,強(qiáng)調(diào)了透過(guò)單一建置過(guò)程連接這些模組的需要。

Gradle 以其多專(zhuān)案建置功能(通常稱(chēng)為多模組專(zhuān)案)來(lái)滿(mǎn)足這種設(shè)定。在 Gradle 的術(shù)語(yǔ)中,這些模組稱(chēng)為子項(xiàng)目。

多專(zhuān)案建置是圍繞一個(gè)根專(zhuān)案建構(gòu)的,並且可以包含其下的多個(gè)子專(zhuān)案。

gradle project

目錄結(jié)構(gòu)應(yīng)如下圖所示:

├── .gradle
│   └── ?
├── gradle
│   ├── libs.versions.toml
│   └── wrapper
├── gradlew
├── gradlew.bat
├── settings.gradle.kts (1)
├── sub-project-1
│   └── build.gradle.kts (2) 
├── sub-project-2
│   └── build.gradle.kts (2) 
└── sub-project-3
    └── build.gradle.kts (2)

(1) settings.gradle.kts 檔案應(yīng)包含所有子項(xiàng)目。
(2) 每個(gè)子項(xiàng)目應(yīng)該有自己的build.gradle.kts檔案。

利用 Gradle 子模組實(shí)現(xiàn)簡(jiǎn)潔的架構(gòu)

清潔架構(gòu)是一種強(qiáng)調(diào)關(guān)注點(diǎn)分離的設(shè)計(jì)模式,使軟體更易於維護(hù)和測(cè)試。在專(zhuān)案中實(shí)作此架構(gòu)的實(shí)用方法之一是使用 Gradle 的子模組結(jié)構(gòu)來(lái)組織程式碼庫(kù)。以下是如何將 Clean Architecture 與 Gradle 子模組結(jié)合:

乾淨(jìng)的架構(gòu)層:
核心:

  • 包含業(yè)務(wù)邏輯、領(lǐng)域模型和應(yīng)用程式規(guī)則。 不依賴(lài)外部或網(wǎng)路。
  • 應(yīng)盡可能獨(dú)立於特定於框架的實(shí)作。

外在:

  • 處理外部操作或集成,例如資料庫(kù)遷移或第三方服務(wù)互動(dòng)。
  • 業(yè)務(wù)邏輯可能依賴(lài) Core,但不應(yīng)依賴(lài) Web。

網(wǎng)頁(yè):

  • 入口點(diǎn),公開(kāi) REST API 並處理 HTTP 請(qǐng)求。
  • 業(yè)務(wù)邏輯依賴(lài)核心,整合可能依賴(lài)外部。
├── .gradle
│   └── ?
├── gradle
│   ├── libs.versions.toml
│   └── wrapper
├── gradlew
├── gradlew.bat
├── settings.gradle.kts (1)
├── sub-project-1
│   └── build.gradle.kts (2) 
├── sub-project-2
│   └── build.gradle.kts (2) 
└── sub-project-3
    └── build.gradle.kts (2)

第 1 步: 建立一個(gè)基於 Java 的 Gradle 專(zhuān)案並將其命名為「SchoolStaff」。

第 2 步: 前往 Spring Initializr 並產(chǎn)生一個(gè)名為 Web.

的 REST API 項(xiàng)目

第 3 步: 建立一個(gè)基於 Java 的 Gradle 專(zhuān)案並命名為 External.

第 4 步: 建立一個(gè)基於 Java 的 Gradle 專(zhuān)案並將其命名為 Core.

根目錄 build.gradle.kts

SchoolStaff/
├── Core/
│   ├── src/
│   │   └── main/
│   │       ├── java/         # Business logic and domain objects
│   │       └── resources/    # Core-specific resources (if any)
│   └── build.gradle.kts
├── External/
│   ├── src/
│   │   └── main/
│   │       ├── java/         # External integration code
│   │       └── resources/    # db/migration and other external resources
│   └── build.gradle.kts
├── Web/
│   ├── src/
│   │   └── main/
│   │       ├── java/         # REST controllers and entry-point logic
│   │       └── resources/    # Application-specific configuration
│   └── build.gradle.kts
├── build.gradle.kts          # Root Gradle build
└── settings.gradle.kts       # Project module settings

settings.gradle.kts

plugins {
    id("java")
}

allprojects {
    group = "school.staff"
    version = "1.0.0"

    repositories {
        mavenLocal()
        mavenCentral()
    }
}

subprojects {
    apply(plugin = "java")

    dependencies {
        testImplementation(platform("org.junit:junit-bom:5.10.0"))
        testImplementation("org.junit.jupiter:junit-jupiter")
    }

    tasks.test {
        useJUnitPlatform()
    }
}

「Web」專(zhuān)案所需的依賴(lài)項(xiàng)。

rootProject.name = "SchoolStaff"

include("Core", "External", "Web")

「核心」專(zhuān)案所需的依賴(lài)項(xiàng)。

dependencies {
    implementation(project(":Core"))
    implementation(project(":External"))
}

「外部」項(xiàng)目所需的依賴(lài)項(xiàng)。

dependencies {
    runtimeOnly(project(":External"))
}

我們使用以下插件進(jìn)行 Flyway 遷移:

import java.sql.DriverManager
import java.util.Properties
// Function to load properties based on the environment
fun loadProperties(env: String): Properties {
    val properties = Properties()
    val propsFile = file("../web/src/main/resources/application-$env.properties")

    if (propsFile.exists()) {
        propsFile.inputStream().use { properties.load(it) }
    } else {
        throw GradleException("Properties file for environment '$env' not found: ${propsFile.absolutePath}")
    }

    return properties
}
// Set the environment (default is 'dev' if no argument is passed)
val env = project.findProperty("env")?.toString() ?: "dev"

// Load properties for the chosen environment
val dbProps = loadProperties(env)
buildscript {
    dependencies {
        classpath("org.flywaydb:flyway-database-postgresql:11.1.0") // This is required for the flyway plugin to work on the migration, otherwise it will throw an error as No Database found
        classpath("org.postgresql:postgresql:42.7.4")
    }
}
plugins {
    id("java-library")
    id("org.flywaydb.flyway") version "11.0.1"
}

group = "school.staff"
version = "unspecified"

repositories {
    mavenLocal()
    mavenCentral()
}

dependencies {
    implementation("org.springframework.boot:spring-boot-starter-data-jpa:3.4.0")
    implementation("org.postgresql:postgresql:42.7.4")
    implementation("org.flywaydb:flyway-core:11.0.1")
    implementation("org.flywaydb:flyway-database-postgresql:11.0.1")
    implementation("org.flywaydb:flyway-gradle-plugin:11.0.1")

    implementation (project(":Core"))
    testImplementation(platform("org.junit:junit-bom:5.10.0"))
    testImplementation("org.junit.jupiter:junit-jupiter")
}

tasks.test {
    useJUnitPlatform()
}

// Task to create the database if it doesn't exist
tasks.register("createDatabase") {
    doLast {
        val dbUrl = dbProps["spring.datasource.url"] as String
        val dbUsername = dbProps["spring.datasource.username"] as String
        val dbPassword = dbProps["spring.datasource.password"] as String

        // Extract the base URL and database name
        val baseDbUrl = dbUrl.substringBeforeLast("/")+ "/"
        val dbName = dbUrl.substringAfterLast("/")

        // Connect to the PostgreSQL server (without the specific database)
        DriverManager.getConnection(baseDbUrl, dbUsername, dbPassword).use { connection ->
            val stmt = connection.createStatement()
            val resultSet = stmt.executeQuery("SELECT 1 FROM pg_database WHERE datname = '$dbName'")
            if (!resultSet.next()) {
                println("Database '$dbName' does not exist. Creating it...")
                stmt.executeUpdate("CREATE DATABASE \"$dbName\"")
                println("Database '$dbName' created successfully.")
            } else {
                println("Database '$dbName' already exists.")
            }
        }
    }
}

flyway {
    url = dbProps["spring.datasource.url"] as String
    user = dbProps["spring.datasource.username"] as String
    password = dbProps["spring.datasource.password"] as String
    locations = arrayOf("classpath:db/migration")
    baselineOnMigrate = true
}

 //Ensure classes are built before migration
tasks.named("flywayMigrate").configure {
    dependsOn(tasks.named("createDatabase"))
    dependsOn(tasks.named("classes"))
}

這種方法非常適合生產(chǎn)環(huán)境,因?yàn)樗_保了受控且可靠的遷移。我們不會(huì)在每次應(yīng)用程式啟動(dòng)時(shí)自動(dòng)運(yùn)行遷移,而是僅在必要時(shí)執(zhí)行遷移,從而提供更大的靈活性和控制力。

我們也利用 Spring 應(yīng)用程式中的 application.properties 檔案來(lái)管理資料庫(kù)連線(xiàn)和憑證。 BaselineOnMigrate = true 設(shè)定可確保將初始遷移用作未來(lái)遷移的基準(zhǔn)。

 plugins {
    id("org.flywaydb.flyway") version "11.0.1"
}

我們可以使用 JPA Buddy 產(chǎn)生外部專(zhuān)案的 resources/db/migration 目錄中的所有遷移檔案。

V1__Initial_Migration

flyway {
    url = dbProps["spring.datasource.url"] as String
    user = dbProps["spring.datasource.username"] as String
    password = dbProps["spring.datasource.password"] as String
    locations = arrayOf("classpath:db/migration")
    baselineOnMigrate = true
}

從根項(xiàng)目中,我們可以使用以下命令執(zhí)行 Flyway 遷移:

CREATE TABLE _user
(
    id                 UUID NOT NULL,
    created_by         UUID,
    created_date       TIMESTAMP WITH TIME ZONE,
    last_modified_by   UUID,
    last_modified_date TIMESTAMP WITH TIME ZONE,
    first_name         VARCHAR(255),
    last_name          VARCHAR(255),
    email              VARCHAR(255),
    password           VARCHAR(255),
    tenant_id          UUID,
    CONSTRAINT pk__user PRIMARY KEY (id)
);

這會(huì)將所有遷移檔案套用到資料庫(kù)。

結(jié)論

我們探索如何在 Gradle 多模組專(zhuān)案中使用 Flyway 自動(dòng)化資料庫(kù)遷移,這對(duì)於在 CI/CD 環(huán)境中維護(hù)架構(gòu)一致性至關(guān)重要。

我們還介紹了 Gradle 如何支援多項(xiàng)目構(gòu)建,將複雜的項(xiàng)目組織成可管理的子項(xiàng)目,每個(gè)子項(xiàng)目都有自己的構(gòu)建配置,統(tǒng)一在根構(gòu)建腳本下。

最後,我們將 Clean Architecture 與 Gradle 模組結(jié)合起來(lái),將專(zhuān)案建置為核心層、外部層和 Web 層,促進(jìn)關(guān)注點(diǎn)和依賴(lài)管理的清晰分離。

這些實(shí)踐增強(qiáng)了模組化、自動(dòng)化和可維護(hù)性,為可擴(kuò)展、無(wú)錯(cuò)誤的軟體開(kāi)發(fā)奠定了基礎(chǔ)。

以上是多模組 Gradle 專(zhuān)案中的 Flyway 遷移(乾淨(jìng)架構(gòu))的詳細(xì)內(nèi)容。更多資訊請(qǐng)關(guān)注PHP中文網(wǎng)其他相關(guān)文章!

本網(wǎng)站聲明
本文內(nèi)容由網(wǎng)友自願(yuàn)投稿,版權(quán)歸原作者所有。本站不承擔(dān)相應(yīng)的法律責(zé)任。如發(fā)現(xiàn)涉嫌抄襲或侵權(quán)的內(nèi)容,請(qǐng)聯(lián)絡(luò)admin@php.cn

熱AI工具

Undress AI Tool

Undress AI Tool

免費(fèi)脫衣圖片

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅(qū)動(dòng)的應(yīng)用程序,用於創(chuàng)建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線(xiàn)上人工智慧工具。

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費(fèi)的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費(fèi)的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強(qiáng)大的PHP整合開(kāi)發(fā)環(huán)境

Dreamweaver CS6

Dreamweaver CS6

視覺(jué)化網(wǎng)頁(yè)開(kāi)發(fā)工具

SublimeText3 Mac版

SublimeText3 Mac版

神級(jí)程式碼編輯軟體(SublimeText3)

熱門(mén)話(huà)題

hashmap和hashtable之間的區(qū)別? hashmap和hashtable之間的區(qū)別? Jun 24, 2025 pm 09:41 PM

HashMap與Hashtable的區(qū)別主要體現(xiàn)在線(xiàn)程安全、null值支持及性能方面。 1.線(xiàn)程安全方面,Hashtable是線(xiàn)程安全的,其方法大多為同步方法,而HashMap不做同步處理,非線(xiàn)程安全;2.null值支持上,HashMap允許一個(gè)null鍵和多個(gè)null值,Hashtable則不允許null鍵或值,否則拋出NullPointerException;3.性能方面,HashMap因無(wú)同步機(jī)制效率更高,Hashtable因每次操作加鎖性能較低,推薦使用ConcurrentHashMap替

為什麼我們需要包裝紙課? 為什麼我們需要包裝紙課? Jun 28, 2025 am 01:01 AM

Java使用包裝類(lèi)是因?yàn)榛緮?shù)據(jù)類(lèi)型無(wú)法直接參與面向?qū)ο癫僮?,而?shí)際需求中常需對(duì)象形式;1.集合類(lèi)只能存儲(chǔ)對(duì)象,如List利用自動(dòng)裝箱存儲(chǔ)數(shù)值;2.泛型不支持基本類(lèi)型,必須使用包裝類(lèi)作為類(lèi)型參數(shù);3.包裝類(lèi)可表示null值,用於區(qū)分未設(shè)置或缺失的數(shù)據(jù);4.包裝類(lèi)提供字符串轉(zhuǎn)換等實(shí)用方法,便於數(shù)據(jù)解析與處理,因此在需要這些特性的場(chǎng)景下,包裝類(lèi)不可或缺。

什麼是接口中的靜態(tài)方法? 什麼是接口中的靜態(tài)方法? Jun 24, 2025 pm 10:57 PM

StaticmethodsininterfaceswereintroducedinJava8toallowutilityfunctionswithintheinterfaceitself.BeforeJava8,suchfunctionsrequiredseparatehelperclasses,leadingtodisorganizedcode.Now,staticmethodsprovidethreekeybenefits:1)theyenableutilitymethodsdirectly

JIT編譯器如何優(yōu)化代碼? JIT編譯器如何優(yōu)化代碼? Jun 24, 2025 pm 10:45 PM

JIT編譯器通過(guò)方法內(nèi)聯(lián)、熱點(diǎn)檢測(cè)與編譯、類(lèi)型推測(cè)與去虛擬化、冗餘操作消除四種方式優(yōu)化代碼。 1.方法內(nèi)聯(lián)減少調(diào)用開(kāi)銷(xiāo),將頻繁調(diào)用的小方法直接插入調(diào)用處;2.熱點(diǎn)檢測(cè)識(shí)別高頻執(zhí)行代碼並集中優(yōu)化,節(jié)省資源;3.類(lèi)型推測(cè)收集運(yùn)行時(shí)類(lèi)型信息實(shí)現(xiàn)去虛擬化調(diào)用,提升效率;4.冗餘操作消除根據(jù)運(yùn)行數(shù)據(jù)刪除無(wú)用計(jì)算和檢查,增強(qiáng)性能。

什麼是實(shí)例初始器塊? 什麼是實(shí)例初始器塊? Jun 25, 2025 pm 12:21 PM

實(shí)例初始化塊在Java中用於在創(chuàng)建對(duì)象時(shí)運(yùn)行初始化邏輯,其執(zhí)行先於構(gòu)造函數(shù)。它適用於多個(gè)構(gòu)造函數(shù)共享初始化代碼、複雜字段初始化或匿名類(lèi)初始化場(chǎng)景,與靜態(tài)初始化塊不同的是它每次實(shí)例化時(shí)都會(huì)執(zhí)行,而靜態(tài)初始化塊僅在類(lèi)加載時(shí)運(yùn)行一次。

什麼是工廠(chǎng)模式? 什麼是工廠(chǎng)模式? Jun 24, 2025 pm 11:29 PM

工廠(chǎng)模式用於封裝對(duì)象創(chuàng)建邏輯,使代碼更靈活、易維護(hù)、松耦合。其核心答案是:通過(guò)集中管理對(duì)象創(chuàng)建邏輯,隱藏實(shí)現(xiàn)細(xì)節(jié),支持多種相關(guān)對(duì)象的創(chuàng)建。具體描述如下:工廠(chǎng)模式將對(duì)象創(chuàng)建交給專(zhuān)門(mén)的工廠(chǎng)類(lèi)或方法處理,避免直接使用newClass();適用於多類(lèi)型相關(guān)對(duì)象創(chuàng)建、創(chuàng)建邏輯可能變化、需隱藏實(shí)現(xiàn)細(xì)節(jié)的場(chǎng)景;例如支付處理器中通過(guò)工廠(chǎng)統(tǒng)一創(chuàng)建Stripe、PayPal等實(shí)例;其實(shí)現(xiàn)包括工廠(chǎng)類(lèi)根據(jù)輸入?yún)?shù)決定返回的對(duì)象,所有對(duì)象實(shí)現(xiàn)共同接口;常見(jiàn)變體有簡(jiǎn)單工廠(chǎng)、工廠(chǎng)方法和抽象工廠(chǎng),分別適用於不同複雜度的需求。

變量的最終關(guān)鍵字是什麼? 變量的最終關(guān)鍵字是什麼? Jun 24, 2025 pm 07:29 PM

InJava,thefinalkeywordpreventsavariable’svaluefrombeingchangedafterassignment,butitsbehaviordiffersforprimitivesandobjectreferences.Forprimitivevariables,finalmakesthevalueconstant,asinfinalintMAX_SPEED=100;wherereassignmentcausesanerror.Forobjectref

什麼是類(lèi)型鑄造? 什麼是類(lèi)型鑄造? Jun 24, 2025 pm 11:09 PM

類(lèi)型轉(zhuǎn)換有兩種:隱式和顯式。 1.隱式轉(zhuǎn)換自動(dòng)發(fā)生,如將int轉(zhuǎn)為double;2.顯式轉(zhuǎn)換需手動(dòng)操作,如使用(int)myDouble。需要類(lèi)型轉(zhuǎn)換的情況包括處理用戶(hù)輸入、數(shù)學(xué)運(yùn)算或函數(shù)間傳遞不同類(lèi)型的值時(shí)。需要注意的問(wèn)題有:浮點(diǎn)數(shù)轉(zhuǎn)整數(shù)會(huì)截?cái)嘈?shù)部分、大類(lèi)型轉(zhuǎn)小類(lèi)型可能導(dǎo)致數(shù)據(jù)丟失、某些語(yǔ)言不允許直接轉(zhuǎn)換特定類(lèi)型。正確理解語(yǔ)言的轉(zhuǎn)換規(guī)則有助於避免錯(cuò)誤。

See all articles