在 Quarkus 的世界中,依賴注入領域豐富且用途廣泛,為開發人員提供了大量管理和控制 Bean 的工具。其中一種工具是合成豆的概念。合成 bean 是一種強大的擴充機制,可讓您註冊其屬性不是從 Java 類別、方法或欄位派生的 bean。相反,合成 bean 的所有屬性都由擴展定義。
在本文中,我們將深入了解 Quarkus 中的合成豆世界。我們將探討合成 bean 的需求、它們的實際應用,以及如何在 Quarkus 應用程式中建立和使用它們。
在 Quarkus 中,bean 是應用程式的構建塊,由上下文和依賴注入 (CDI) 框架管理。通常,CDI bean 是使用各種 CDI 註解(例如 @ApplicationScoped、@RequestScoped 或 @Inject)進行註解的 Java 類別。這些註釋
允許 CDI 自動管理 Bean 的生命週期和注入。
但是,在某些情況下,您可能需要註冊一個不太適合傳統 CDI 模型的 Bean。這就是合成豆發揮作用的地方。合成 bean 由擴充創建,其屬性完全由這些擴充定義。在常規 CDI 的世界中,您可以使用 AfterBeanDiscovery.addBean() 和 SyntheticComponents.addBean() 方法來實現此目的。在 Quarkus 中,這是使用 SyntheticBeanBuildItem 完成的。
那麼,什麼時候您可能需要在 Quarkus 中使用合成 Bean?合成豆在以下情況下是一個強大的工具:
整合第三方程式庫:您正在使用沒有 CDI 註解但需要整合到基於 CDI 的應用程式中的第三方程式庫。合成豆可以幫助您彌補這一差距。
動態 Bean 註冊: 您需要在運行時動態註冊 Bean,具體取決於配置或其他因素。合成 Bean 讓您可以靈活地動態建立和註冊 Bean。
自訂 Bean 管理:您需要對 Bean 的範圍和行為進行細微控制,這是標準 CDI 註釋無法實現的。
實作專用 Bean:您想要建立具有與傳統 Java 類別或方法不對應的獨特屬性的專用 Bean。
模擬依賴項以進行測試:合成 bean 提供了一種有用的方法來模擬依賴項並注入模擬實作以進行測試。
SynthesisFinishedBuildItem 用於指示 CDI bean 發現和註冊過程已完成。這允許擴充功能知道何時可以安全地與已註冊的 Bean 進行互動。
例如:
@BuildStep void onSynthesisFinished(SynthesisFinishedBuildItem synthesisFinished){ // CDI bean registration is complete, can now safely interact with beans }
SyntheticBeansRuntimeInitBuildItem 用於註冊一個回調,該回調將在所有合成 bean 初始化後在執行時呼叫。如果您需要執行涉及合成 bean 的額外初始化邏輯,這非常有用。
例如:
@BuildStep SyntheticBeansRuntimeInitBuildItem initSyntheticBeans(){ return new SyntheticBeansRuntimeInitBuildItem(ids->{ // Perform logic with initialized synthetic beans }); }
傳遞給 SyntheticBeansRuntimeInitBuildItem 的回呼將會收到一個 Set
總而言之,SynthesisFinishedBuildItem 表示 Bean 發現已完成,而 SyntheticBeansRuntimeInitBuildItem 允許根據合成 Bean 初始化邏輯。
在 Quarkus 中,建立合成 bean 是一個簡單的過程,這要歸功於 SyntheticBeanBuildItem 類別。讓我們逐步完成建立和使用合成 bean 的步驟:
package com.iqnev; public class MySyntheticBean { // Define the behavior and attributes of your synthetic bean public void printMessage() { System.out.println("Hello from synthetic bean!"); } }
package com.iqnev; import io.quarkus.arc.deployment.SyntheticBeanBuildItem; public class MySyntheticBeanExtension { @BuildStep SyntheticBeanBuildItem syntheticBean() { return SyntheticBeanBuildItem .configure(MySyntheticBean.class) .scope(ApplicationScoped.class) .creator(mc -> { mc.returnValue(new MySyntheticBean()); }) .done(); } }
SyntheticBeanBuildItem 上的 .creator() 方法用於產生字節碼,該字節碼將在執行時建立合成 bean 的實例。
傳遞給 .creator() 的參數是 Consumer
在此範例中:
So essentially, we are telling Quarkus to generate a method that looks something like:
MySyntheticBean createSyntheticBean(){ return new MySyntheticBean(); }
This generated method will then be called to instantiate the MySyntheticBean when it needs to be injected or used.
The reason bytecode generation is used is that synthetic beans do not correspond to real Java classes/methods, so we have to explicitly generate a method to instantiate them
The output of SyntheticBeanBuildItem is bytecode recorded at build time. This limits how instances are created at runtime. Common options are:
The @Record and .runtimeValue() approaches are alternate ways of providing instances for synthetic beans in Quarkus.
This allows you to instantiate the synthetic bean via a recorder class method annotated with @Record(STATIC_INIT).
For example:
@Recorder public class MyRecorder { @Record(STATIC_INIT) public MySyntheticBean createBean() { return new MySyntheticBean(); } } @BuildStep SyntheticBeanBuildItem syntheticBean(MyRecorder recorder) { return SyntheticBeanBuildItem .configure(MySyntheticBean.class) .runtimeValue(recorder.createBean()); }
Here the .runtimeValue() references the recorder method to instantiate the bean. This allows passing a RuntimeValue directly to provide the synthetic bean instance.
For example:
@BuildStep SyntheticBeanBuildItem syntheticBean(){ RuntimeValue<MySyntheticBean> bean= //... return SyntheticBeanBuildItem .configure(MySyntheticBean.class) .runtimeValue(bean); }
The RuntimeValue could come from a recorder, supplier, proxy etc.
So in summary:
They both achieve the same goal of providing a runtime instance, just in slightly different ways.
When it comes to providing runtime instances for synthetic beans in Quarkus, I would consider using recorders (via @Record) to be a more advanced approach compared to directly generating bytecode
with .creator() or supplying simple RuntimeValues.
Here are some reasons why using recorders can be more advanced:
So in summary, recorder methods provide more encapsulation, flexibility and access to runtime data and services for instantiating synthetic beans. They allow for more advanced bean production logic compared to direct bytecode generation.
However, direct bytecode generation with .creator() can still be useful for simple cases where recorders may be overkill. But as synthetic bean needs grow, recorders are a more powerful and
advanced approach.
It is possible to configure a synthetic bean in Quarkus to be initialized during the RUNTIME_INIT phase instead of the default STATIC_INIT phase.
Here is an example:
@BuildStep @Record(RUNTIME_INIT) SyntheticBeanBuildItem lazyBean(BeanRecorder recorder){ return SyntheticBeanBuildItem .configure(MyLazyBean.class) .setRuntimeInit() // initialize during RUNTIME_INIT .runtimeValue(recorder.createLazyBean()); }
The key points are:
So in summary, synthetic beans can be initialized lazily during RUNTIME_INIT for cases where eager STATIC_INIT instantiation is not needed. This allows optimizing startup time.
Use the Synthetic Bean: Now that your synthetic bean is registered, you can inject and use it in your application.
package com.iqnev; import javax.inject.Inject; public class MyBeanUser { @Inject MySyntheticBean mySyntheticBean; public void useSyntheticBean() { // Use the synthetic bean in your code mySyntheticBean.printMessage(); } }
Running Your Application: Build and run your Quarkus application as usual, and the synthetic bean will be available for injection and use.
Synthetic beans in Quarkus provide a powerful mechanism for integrating external libraries, dynamically registering beans, and customizing bean behavior in your CDI-based applications. These beans, whose attributes are defined by extensions rather than Java classes, offer flexibility and versatility in managing dependencies.
正如我們在本文中探討的那樣,在 Quarkus 中創建和使用合成 bean 是一個簡單的過程。透過利用 SyntheticBeanBuildItem 和 Quarkus 擴展,您可以無縫地彌合傳統 CDI 與更專業或動態 bean 註冊要求之間的差距。
在不斷發展的 Java 框架領域,Quarkus 透過提供合成 Bean 等創新解決方案繼續脫穎而出,使其成為現代、高效和靈活的應用程式開發的引人注目的選擇。擁抱 Quarkus 中合成 bean 的強大功能,將您的依賴注入提升到一個新的水平!
以上是探索 Quarkus 中的合成豆。強大的擴充機制的詳細內容。更多資訊請關注PHP中文網其他相關文章!