Springboot遷移到Micronaut實現過程詳解
設置上下文
JVM是一項偉大的技術。現代版本將運行的字節碼編譯為本機代碼,具體取決於現有的工作負載。出於這個原因,JVM 應用程序在運行時性能方面與本機可執行文件不相上下,甚至勝過本機可執行文件。
JVM 應用程序有一個預熱時間,在此期間它們性能不佳。在運行時加載類沒有幫助。Spring 和 Jakarta EE 等框架一直在使用類路徑掃描和反射,這使得啟動時間更長。對於長時間運行的進程,例如傳統的應用程序服務器,這不是問題。
在容器的上下文中,它是。因為人們把容器當作牛而不是寵物來處理,所以 Kubernetes等平臺會定期殺死 pod 並安排新的 pod。啟動時間越長,JVM 的相關性就越低。在需要快速自動縮放 pod 數量的無服務器環境中,情況會變得更糟。
為瞭趕上潮流,Oracle 提供瞭SubstrateVM。GraalVM 的子組件 SubstrateVM 允許將 JVM 字節碼轉換為本機可執行文件。為此,SubstrateVM 編譯字節碼AOT。出於這個原因,您需要在運行時顯式地向它提供 JVM 上可用的信息。例如反射的情況。請註意,某些 JVM 功能未移植到 GraalVM。此外,AOT編譯是一個耗時的過程。
結果是,一方面,我們擁有 JVM 及其框架所利用的所有功能;另一方面,我們有本機可執行文件,需要精細的手動配置和大量的構建時間。
新一代的框架已經產生,旨在找到一個中間地帶 ,即Micronaut 和 Quarkus。它們都旨在生成字節碼 AOT。請註意,此 AOT 與上面提到的不同。兩個框架都沒有在運行時使用昂貴的反射,而是在構建時生成額外的類。這也使我們能夠避免在啟動時進行類路徑掃描。簡而言之,這個想法是關於在構建時提供盡可能多的代碼。
示例應用程序
我希望遷移的示例應用程序足夠簡單,這樣我就可以自己遷移它,但又不至於變得微不足道。它由以下部分組成:
- Spring MVC實現的控制器層
- 由 Spring Data JPA 實現的存儲庫層
- 一個 JPA 實體
- 通過 Spring Boot 在啟動時生成模式和數據插入
- Spring Boot 執行器,啟用瞭
health
和beans
端點,無需身份驗證即可訪問
該應用程序是用 Kotlin 編寫的。我將使用 H2 作為數據庫來簡化整個設置。
常見變化
第一個變化是替換父 POM。
<parent> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-parent</artifactId> <version>2.3.5.RELEASE</version> <relativePath/> <!-- lookup parent from repository --> </parent> <parent> <groupId>io.micronaut</groupId> <artifactId>micronaut-parent</artifactId> <version>2.1.3</version> </parent>
因為 Micronaut 在構建時生成字節碼,所以我們需要在編譯期間添加註釋處理器。因此,最後的第二步是在 POM 中配置它。
<plugin> <groupId>org.jetbrains.kotlin</groupId> <artifactId>kotlin-maven-plugin</artifactId> <version>${kotlin.version}</version> ... <executions> <execution> <id>kapt</id> <goals> <goal>kapt</goal> </goals> <configuration> <annotationProcessorPaths> <annotationProcessorPath> <groupId>io.micronaut</groupId> <artifactId>micronaut-inject-java</artifactId> <!-- 1 --> <version>${micronaut.version}</version> </annotationProcessorPath> <annotationProcessorPath> <groupId>io.micronaut.data</groupId> <artifactId>micronaut-data-processor</artifactId> <!-- 2 --> <version>${micronaut.data.version}</version> </annotationProcessorPath> </annotationProcessorPaths> </configuration> </execution> ... </executions> ... </plugin>
- 處理依賴註入
- 處理持久化相關的類
您可以通過查看target/classes
文件夾來檢查那些額外的類。例如,示例應用程序顯示以下內容:
$Person$Introspection$$0.class PersonRepository$Intercepted$$proxy0.class
$Person$Introspection$$1.class PersonRepository$Intercepted$$proxy1.class
$Person$Introspection$$2.class PersonRepository$Intercepted$$proxy10.clas
$Person$Introspection$$3.class PersonRepository$Intercepted$$proxy2.class
$Person$Introspection.class PersonRepository$Intercepted$$proxy3.class
$Person$IntrospectionRef.class PersonRepository$Intercepted$$proxy4.class
$PersonControllerDefinition$$exec1.class PersonRepository$Intercepted$$proxy5.class
$PersonControllerDefinition$$exec2.class PersonRepository$Intercepted$$proxy6.class
$PersonControllerDefinition.class PersonRepository$Intercepted$$proxy7.class
$PersonControllerDefinitionClass.class PersonRepository$Intercepted$$proxy8.class
$PersonRepository$InterceptedDefinition.class PersonRepository$Intercepted$$proxy9.class
$PersonRepository$InterceptedDefinitionClass.class PersonRepository$Intercepted.class
Person.class PersonRepository.class
PersonController.class SpringToMicronautApplicationKt.class
Micronaut 創建包含Introspection
並Intercepted
通過kapt
.
為瞭啟動應用程序,Spring Boot 引用瞭一個類。
@SpringBootApplication class SpringToMicronautApplication fun main(args: Array<String>) { runApplication<SpringToMicronautApplication>(*args) }
Micronaut 允許我們隻使用標準main
函數。
fun main(args: Array<String>) { build() .args(*args) .packages("ch.frankel.springtomicronaut") .start() }
Spring Boot 插件可以main
“自動”找到函數。在 Micronaut 中,當前版本要求您在 POM 中顯式設置它:
<properties> ... <exec.mainClass>ch.frankel.s2m.SpringToMicronautApplicationKt</exec.mainClass> </properties>
遷移 web 層
遷移到 web 層需要:
- 用相關的 Micronaut 依賴項替換 Spring Boot 啟動器
- 用 Micronaut 的註釋替換 Spring Boot 的註釋
為瞭使應用程序成為 Web 應用程序,Micronaut 要求添加嵌入式服務器依賴項。Tomcat、Jetty 和 Undertow 可用。由於Spring Boot默認是Tomcat,所以我們使用Tomcat:
<dependency> <groupId>io.micronaut.servlet</groupId> <artifactId>micronaut-http-server-tomcat</artifactId> <scope>runtime</scope> </dependency>
Spring 和 Micronaut 的註解幾乎是一一對應的。使用 Micronaut 隻是使用一個包的註釋而不是另一個包的註釋的問題。Controller
不同之處在於 Spring 提供瞭使用專門的註解將序列化為 JSON 的能力, @RestController
. Micronaut 不需要也不需要在Controller
註解上設置屬性。
春天 | 微航 |
---|---|
o.s.w.b.a.RestController | i.m.h.a.Controller(produces = [TEXT_JSON]) |
o.s.w.b.a.GetMapping | i.m.h.a.Get |
o.s.w.b.a.PathVariable | i.m.h.a.PathVariable |
o.s.w.b.a=org.springframework.web.bind.annotation
i.m.h.a=io.micronaut.http.annotation
遷移數據訪問層
要遷移到數據訪問層,必須:
- 使用 Micronaut 的依賴項而不是 Spring Boot 的
- 將 Micronaut 的 Spring Boot 替換
Repository
為 Micronaut 的 - 使用 Micronaut 創建模式並加載初始數據
要創建數據源和連接池,Spring Boot 需要一個 Spring Data starter 和相關的驅動程序。Micronaut 需要三個不同的部分:
- 數據訪問依賴
- 驅動程序依賴
- 連接池依賴
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-jpa</artifactId> </dependency> <dependency> <groupId>io.micronaut.data</groupId> <artifactId>micronaut-data-hibernate-jpa</artifactId> <version>${micronaut.data.version}</version> </dependency> <dependency> <groupId>io.micronaut.sql</groupId> <artifactId>micronaut-jdbc-hikari</artifactId> </dependency>
請註意,如果您忘記瞭連接池,您將在運行時遇到此錯誤:
No backing RepositoryOperations configured for repository. Check your configuration and try again
Spring Data JPA 在運行時生成存儲庫的實現。Micronaut Data 在構建時生成它們。對於開發人員來說,主要區別在於存儲庫接口必須使用 Micronaut 的@Repository
.
@Repository interface PersonRepository : CrudRepository<Person, Long>
需要配置 Micronaut 以掃描存儲庫和實體:
jpa.default: packages-to-scan: - 'ch.frankel.springtomicronaut'
要創建模式,您可以通過兩種不同的方式配置 Spring Boot:依賴 Hibernate 的模式創建或create.sql
在類路徑的根目錄中提供一個文件。同樣,要插入初始數據,您可以添加一個data.sql
.
Micronaut 不提供開箱即用的機制來插入數據。但它提供瞭與 Flyway 的集成。放置 Flyway 的遷移的默認位置是db/migration
,就像 Spring Boot 一樣。
<dependency> <groupId>io.micronaut.flyway</groupId> <artifactId>micronaut-flyway</artifactId> <version>2.1.1</version> </dependency>
jpa.default: properties.hibernate: hbm2ddl.auto: none # 1 show_sql: true # 2 flyway.datasources.default: enabled # 3
- 禁用 Hibernate 的模式創建
- 記錄 SQL 語句
- 啟用 Flyway 遷移
H2 驅動程序依賴性保持不變。雖然 Spring Boot 使用默認參數創建連接,但 Micronaut 需要顯式配置它:
datasources.default: url: jdbc:h2:mem:test driverClassName: org.h2.Driver username: sa dialect: H2
遷移執行器
Micronaut 還提供管理端點。它與 Spring Boot 的基本相同。
需要替換依賴項:
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-actuator</artifactId> </dependency> <dependency> <groupId>io.micronaut</groupId> <artifactId>micronaut-management</artifactId> </dependency>
與 Spring Boot 最大的區別是開發人員需要單獨配置端點:
endpoints: all.path: /actuator # 1 beans: enabled: true sensitive: false health: enabled: true sensitive: false flyway: enabled: true sensitive: false
這樣 一個Springboot項目遷移到Micronaut項目就完成啦!
以上就是Springboot遷移到Micronaut實現過程詳解的詳細內容,更多關於Springboot遷移Micronaut的資料請關註WalkonNet其它相關文章!
推薦閱讀:
- 使用maven開發springboot項目時pom.xml常用配置(推薦)
- SpringBoot的依賴管理配置
- Springboot基礎學習之初識SpringBoot
- springboot集成camunda的實現示例
- SpringBoot快速遷移至Quarkus的方法步驟