Migrasi Flyway dalam Projek Gradle Berbilang Modul (Seni Bina Bersih)
Jan 19, 2025 am 08:05 AMMengautomasikan Migrasi Pangkalan Data di Java dengan Flyway
Penghijrahan pangkalan data ialah aspek penting dalam pembangunan perisian, terutamanya dalam persekitaran di mana penyepaduan dan penghantaran berterusan (CI/CD) adalah amalan standard. Apabila aplikasi anda berkembang dan berkembang, skema pangkalan data juga mesti bergantung padanya. Menguruskan perubahan skema ini secara manual boleh menyebabkan ralat dan memakan masa yang ketara.
Enter Flyway, alat sumber terbuka yang tidak ternilai yang disesuaikan untuk memudahkan migrasi pangkalan data. Flyway memperkenalkan kawalan versi ke pangkalan data anda, membolehkan anda memindahkan skema anda dengan selamat dan dengan kebolehpercayaan. Dalam artikel ini, kami akan meneroka cara untuk mengautomasikan pemindahan pangkalan data dalam projek gragle java berbilang modul menggunakan Flyway, memastikan pengurusan perubahan pangkalan data menjadi proses yang diperkemas dan tahan ralat.
Maklumat lanjut tentang flyway
Memahami Binaan Berbilang Projek dalam Gradle
Walaupun sesetengah projek yang lebih kecil atau aplikasi monolitik mungkin diuruskan dengan hanya satu fail binaan dan struktur sumber bersatu, projek yang lebih besar kerap disusun menjadi beberapa modul yang saling bergantung. Istilah "saling bergantung" adalah penting di sini, menonjolkan keperluan untuk menyambungkan modul ini melalui proses binaan tunggal.
Gradle memenuhi persediaan ini dengan keupayaan binaan berbilang projeknya, yang sering disebut sebagai projek berbilang modul. Dalam terminologi Gradle, modul ini dipanggil subprojek.
Binaan berbilang projek distrukturkan di sekitar satu projek akar dan boleh termasuk beberapa subprojek di bawahnya.
Struktur direktori hendaklah kelihatan seperti berikut:
├── .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) Fail settings.gradle.kts hendaklah termasuk semua subprojek.
(2) Setiap subprojek harus mempunyai fail build.gradle.kts sendiri.
Memanfaatkan Sub-Modul Gradle untuk Seni Bina Bersih
Clean Architecture ialah corak reka bentuk yang menekankan pengasingan kebimbangan, menjadikan perisian lebih mudah untuk diselenggara dan diuji. Salah satu cara praktikal untuk melaksanakan seni bina ini dalam projek melibatkan penggunaan struktur submodul Gradle untuk menyusun pangkalan kod anda. Begini cara anda boleh menyelaraskan Seni Bina Bersih dengan sub-modul Gradle:
Lapisan Seni Bina Bersih:
Teras:
- Mengandungi logik perniagaan, model domain dan peraturan aplikasi. Tidak mempunyai pergantungan pada Luaran atau Web.
- Hendaklah bebas daripada pelaksanaan khusus rangka kerja jika boleh.
Luaran:
- Mengendalikan tindakan atau penyepaduan luaran, seperti pemindahan pangkalan data atau interaksi perkhidmatan pihak ketiga.
- Mungkin bergantung pada Teras untuk logik perniagaan tetapi tidak boleh bergantung pada Web.
Web:
- Titik masuk, mendedahkan REST API dan mengendalikan permintaan HTTP.
- Bergantung pada Teras untuk logik perniagaan dan mungkin bergantung pada Luaran untuk penyepaduan.
├── .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)
Langkah 1: Buat projek Gradle berasaskan Java dan namakannya "SchoolStaff".
Langkah 2: Pergi ke Spring Initializr dan hasilkan projek REST API bernama Web.
Langkah 3: Cipta projek Gradle berasaskan Java dan namakannya Luaran.
Langkah 4: Buat projek Gradle berasaskan Java dan namakannya Teras.
Root 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() } }
Kebergantungan yang diperlukan untuk projek "Web".
rootProject.name = "SchoolStaff" include("Core", "External", "Web")
Kebergantungan yang diperlukan untuk projek "Teras".
dependencies { implementation(project(":Core")) implementation(project(":External")) }
Kebergantungan yang diperlukan untuk projek "Luaran".
dependencies { runtimeOnly(project(":External")) }
Kami menggunakan pemalam berikut untuk penghijrahan 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")) }
Pendekatan ini sangat sesuai untuk persekitaran pengeluaran, kerana ia memastikan migrasi terkawal dan boleh dipercayai. Daripada menjalankan migrasi secara automatik pada setiap permulaan aplikasi, kami melaksanakannya hanya apabila perlu, memberikan fleksibiliti dan kawalan yang lebih besar.
Kami juga menggunakan fail application.properties dalam aplikasi Spring untuk mengurus sambungan pangkalan data dan bukti kelayakan. Tetapan baselineOnMigrate = benar memastikan bahawa migrasi awal digunakan sebagai garis dasar untuk migrasi masa hadapan.
plugins { id("org.flywaydb.flyway") version "11.0.1" }
Kami boleh menggunakan JPA Buddy untuk menjana semua fail migrasi dalam direktori sumber/db/migrasi projek Luaran.
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 }
Daripada projek root, kita boleh melaksanakan migrasi Flyway menggunakan arahan berikut:
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) );
Ini akan menggunakan semua fail migrasi ke pangkalan data.
Kesimpulan
Kami telah meneroka cara mengautomasikan migrasi pangkalan data menggunakan Flyway dalam projek berbilang modul Gradle, yang penting untuk mengekalkan ketekalan skema dalam persekitaran CI/CD.
Kami juga membincangkan cara Gradle menyokong binaan berbilang projek, mengatur projek yang kompleks menjadi subprojek yang boleh diurus, masing-masing dengan konfigurasi binaan sendiri, disatukan di bawah skrip binaan akar.
Akhir sekali, kami menyelaraskan Seni Bina Bersih dengan modul Gradle, menstrukturkan projek ke dalam lapisan Teras, Luaran dan Web, mempromosikan pengasingan bersih kebimbangan dan pengurusan pergantungan.
Amalan ini meningkatkan kemodulatan, automasi dan kebolehselenggaraan, menetapkan peringkat untuk pembangunan perisian bebas ralat berskala.
Atas ialah kandungan terperinci Migrasi Flyway dalam Projek Gradle Berbilang Modul (Seni Bina Bersih). Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

Alat AI Hot

Undress AI Tool
Gambar buka pakaian secara percuma

Undresser.AI Undress
Apl berkuasa AI untuk mencipta foto bogel yang realistik

AI Clothes Remover
Alat AI dalam talian untuk mengeluarkan pakaian daripada foto.

Clothoff.io
Penyingkiran pakaian AI

Video Face Swap
Tukar muka dalam mana-mana video dengan mudah menggunakan alat tukar muka AI percuma kami!

Artikel Panas

Alat panas

Notepad++7.3.1
Editor kod yang mudah digunakan dan percuma

SublimeText3 versi Cina
Versi Cina, sangat mudah digunakan

Hantar Studio 13.0.1
Persekitaran pembangunan bersepadu PHP yang berkuasa

Dreamweaver CS6
Alat pembangunan web visual

SublimeText3 versi Mac
Perisian penyuntingan kod peringkat Tuhan (SublimeText3)

Topik panas

Perbezaan antara hashmap dan hashtable terutamanya dicerminkan dalam keselamatan benang, sokongan nilai null dan prestasi. 1. Dari segi keselamatan benang, hashtable adalah benang selamat, dan kaedahnya kebanyakannya kaedah segerak, sementara hashmap tidak melakukan pemprosesan penyegerakan, yang bukan benang-selamat; 2. Dari segi sokongan nilai null, hashmap membolehkan satu kunci null dan nilai null berbilang, manakala hashtable tidak membenarkan kekunci atau nilai null, jika tidak, nullPointerException akan dibuang; 3. Dari segi prestasi, hashmap lebih cekap kerana tidak ada mekanisme penyegerakan, dan Hashtable mempunyai prestasi penguncian yang rendah untuk setiap operasi. Adalah disyorkan untuk menggunakan ConcurrentHashMap sebaliknya.

Java menggunakan kelas pembalut kerana jenis data asas tidak dapat mengambil bahagian secara langsung dalam operasi berorientasikan objek, dan bentuk objek sering diperlukan dalam keperluan sebenar; 1. Kelas koleksi hanya boleh menyimpan objek, seperti senarai menggunakan tinju automatik untuk menyimpan nilai berangka; 2. Generik tidak menyokong jenis asas, dan kelas pembungkusan mesti digunakan sebagai parameter jenis; 3. Kelas pembungkusan boleh mewakili nilai null untuk membezakan data yang tidak tersendiri atau hilang; 4. Kelas pembungkusan menyediakan kaedah praktikal seperti penukaran rentetan untuk memudahkan parsing dan pemprosesan data, jadi dalam senario di mana ciri -ciri ini diperlukan, kelas pembungkusan sangat diperlukan.

Staticmethodsininterfaceswereintroducedinjava8toallowutilityfunctionswithintheintheinterfaceitself.beforjava8, SuchfunctionsRequiredseparateHelpereHelperes, LeadingTodisorgaganizedCode.Now, staticmethodethreeKeybeeMeKeBeReSes, staticmethodeDethreeKeybeeMeKeBeReSes, staticmethodethreeKeybeeMeKeKeBeReSes, staticmethodeDethreeKeybeeMeKeKeBeReKeNey

Penyusun JIT mengoptimumkan kod melalui empat kaedah: kaedah dalam talian, pengesanan tempat panas dan penyusunan, spekulasi jenis dan devirtualisasi, dan penghapusan operasi yang berlebihan. 1. Kaedah sebaris mengurangkan panggilan overhead dan memasukkan kaedah kecil yang sering dipanggil terus ke dalam panggilan; 2. Pengesanan tempat panas dan pelaksanaan kod frekuensi tinggi dan mengoptimumkannya untuk menjimatkan sumber; 3. Jenis spekulasi mengumpul maklumat jenis runtime untuk mencapai panggilan devirtualisasi, meningkatkan kecekapan; 4. Operasi berlebihan menghapuskan pengiraan dan pemeriksaan yang tidak berguna berdasarkan penghapusan data operasi, meningkatkan prestasi.

Blok permulaan contoh digunakan dalam Java untuk menjalankan logik inisialisasi apabila membuat objek, yang dilaksanakan sebelum pembina. Ia sesuai untuk senario di mana beberapa pembina berkongsi kod inisialisasi, permulaan medan kompleks, atau senario permulaan kelas tanpa nama. Tidak seperti blok inisialisasi statik, ia dilaksanakan setiap kali ia ditegaskan, manakala blok permulaan statik hanya dijalankan sekali apabila kelas dimuatkan.

Mod kilang digunakan untuk merangkum logik penciptaan objek, menjadikan kod lebih fleksibel, mudah dikekalkan, dan ditambah longgar. Jawapan teras adalah: dengan mengurus logik penciptaan objek secara berpusat, menyembunyikan butiran pelaksanaan, dan menyokong penciptaan pelbagai objek yang berkaitan. Keterangan khusus adalah seperti berikut: Mod Kilang menyerahkan penciptaan objek ke kelas kilang khas atau kaedah untuk diproses, mengelakkan penggunaan Newclass () secara langsung; Ia sesuai untuk senario di mana pelbagai jenis objek yang berkaitan dicipta, logik penciptaan boleh berubah, dan butiran pelaksanaan perlu disembunyikan; Sebagai contoh, dalam pemproses pembayaran, jalur, paypal dan contoh lain dicipta melalui kilang -kilang; Pelaksanaannya termasuk objek yang dikembalikan oleh kelas kilang berdasarkan parameter input, dan semua objek menyedari antara muka yang sama; Varian biasa termasuk kilang -kilang mudah, kaedah kilang dan kilang abstrak, yang sesuai untuk kerumitan yang berbeza.

Injava, thefinalkeywordpreventsavariable'svaluefrombeingchangedafterassignment, butitsbehaviordiffersforprimitivesandobjectreferences.forprimitiveVariables, finalmakesthevalueconstant, asinfinalintmax_speed = 100;

Terdapat dua jenis penukaran: tersirat dan eksplisit. 1. Penukaran tersirat berlaku secara automatik, seperti menukar int untuk berganda; 2. Penukaran eksplisit memerlukan operasi manual, seperti menggunakan (int) mydouble. Kes di mana penukaran jenis diperlukan termasuk memproses input pengguna, operasi matematik, atau lulus pelbagai jenis nilai antara fungsi. Isu-isu yang perlu diperhatikan adalah: Mengubah nombor terapung ke dalam bilangan bulat akan memotong bahagian pecahan, mengubah jenis besar menjadi jenis kecil boleh menyebabkan kehilangan data, dan beberapa bahasa tidak membenarkan penukaran langsung jenis tertentu. Pemahaman yang betul tentang peraturan penukaran bahasa membantu mengelakkan kesilapan.
