Spring IOC官方文檔學習筆記(十二)之基於Java的容器配置

来源:https://www.cnblogs.com/shame11/archive/2023/02/22/17133937.html
-Advertisement-
Play Games

安裝apk 安裝APK(filePath) lua源碼 function 安裝APK(filePath) local intent = Intent(Intent.ACTION_VIEW); intent.addCategory("android.intent.category.DEFAULT"); ...


1.@Bean與@Configuration

(1) 標註於類之上的@Configuration註解與標註於方法之上的@Bean註解是支持基於Java的容器配置的核心,被@Bean註解標註的方法用於實例化bean並將其註入至容器中,它與基於xml配置中的<bean/>標簽起著相同的作用,@Bean可用在任何被@Component註解標註的類中,不過絕大部分情況下它們都被用於被@Configuration註解標註的類中;被@Configuration註解標註的類通常作為bean的定義源,如同基於xml配置中的<beans/>標簽,此外,還可在@Configuration標註的類中配置bean之間的依賴關係,如下

//兩個普通的類,其中ExampleB依賴ExampleA
public class ExampleA { }

public class ExampleB {

    private ExampleA exampleA;

    public ExampleB(ExampleA exampleA) {
        this.exampleA = exampleA;
    }
}

//配置類
@Configuration
public class Config {
    //註入bean ExampleA
    @Bean
    public ExampleA exampleA() {
        return new ExampleA();
    }

    //調用exampleA()方法來配置ExampleB
    @Bean
    public ExampleB exampleB() {
        return new ExampleB(exampleA());
    }

    //上下這兩個exampleB方法等價,註意,在下麵這個例子中,Spring會自動幫我們註入ExampleA對象
//    @Bean
//    public ExampleB exampleB(ExampleA exampleA) {
//        return new ExampleB(exampleA);
//    }
}

//啟動容器,列印註入的對象是否相同
AnnotationConfigApplicationContext ctx = new AnnotationConfigApplicationContext(Config.class);
System.out.println(ctx.getBean(ExampleB.class).getExampleA() == ctx.getBean(ExampleA.class));

//觀察結果,可見Spring執行了依賴註入,註入了容器中的ExampleA,而非new出來了一個新的ExampleA
true

(2) @Bean也可用在任何被@Component註解標註的類中,此時,我們稱其為lite @Bean模式,而在這種lite模式下,我們不能配置bean之間的依賴關係,如下所示

//取上面的例子,其他保持不變,只將Config類上的@Configuration註解變更為@Component註解,此時其中的@Bean註解就處於lite @Bean模式
@Component
public class Config {

    //保持不變...
}

//啟動容器,觀察列印結果,為false,可見此時Spring並沒有執行依賴註入,而是直接new出來了一個新的ExampleA給ExampleB,因此在這種lite模式下,我們不能配置bean之間的依賴關係

2.通過AnnotationConfigApplicationContext實例化Spring容器

(1) AnnotationConfigApplicationContext作為Spring的容器,它不僅可以接受@Configuration類(同時這個類本身也會被註冊為一個bean)作為參數,還可以接受@Component類或用JSR-330註解標註的類作為參數,如下

//例一: 一個Spring配置類
@Configuration
public class Config {

}

public static void main(String[] args) {
    //使用@Configuration類作為輸入,完全擺脫掉基於xml的配置  
    AnnotationConfigApplicationContext ctx = new AnnotationConfigApplicationContext(Config.class);
    //列印,會觀察到Config也被註入到了容器中
    Arrays.stream(ctx.getBeanDefinitionNames()).forEach(System.out::println);
}

//例二: ExampleB被JSR 330標準註解標註
@Component
public class ExampleA {

}

@Named
public class ExampleB {
    @Inject
    private ExampleA exampleA;

    public ExampleA getExampleA() {
        return exampleA;
    }
}

public static void main(String[] args) {
    //使用@Component類或用JSR-330註解標註的類作為參數,容器會對它們進行依賴註入
    AnnotationConfigApplicationContext ctx = new AnnotationConfigApplicationContext(ExampleA.class, ExampleB.class);
    System.out.println(ctx.getBean(ExampleB.class).getExampleA());
}

(2) 可以使用AnnotationConfigApplicationContext類的register(Class<?>…​)方法編程式的向容器中註冊,如下

public class ExampleA { }

public static void main(String[] args) {
    //使用無參構造函數
    AnnotationConfigApplicationContext ctx = new AnnotationConfigApplicationContext();
    //編程式的向容器中註冊bean ExampleA
    ctx.register(ExampleA.class);
    //refresh()方法用於處理bean,一定要調用此方法,否則容器將拋出異常
    ctx.refresh();
    System.out.println(ctx.getBean(ExampleA.class));
}

(3) 我們可以使用@ComponentScan(basePackages = "...")註解或<context:component-scan base-package="..."/>標簽來開啟註解掃描,此外,AnnotationConfigApplicationContext也提供了scan(String…​)方法來開啟註解掃描,如下

//ExampleA位於cn.example.spring.boke包下
@Component
public class ExampleA { }

public static void main(String[] args) {
    AnnotationConfigApplicationContext ctx = new AnnotationConfigApplicationContext();
    //開啟註解掃描,掃描cn.example.spring.boke"包下的bean
    ctx.scan("cn.example.spring.boke");
    ctx.refresh();
    System.out.println(ctx.getBean(ExampleA.class));
}

(4) 在web環境中,可以使用AnnotationConfigWebApplicationContext,來配置ContextLoaderListener監聽器或DispatcherServlet,具體配置細節可參考官方文檔

3.使用@Bean註解

(1) @Bean註解是一個方法級別的註解,類似於基於xml配置中的<bean/>標簽,我們可以在@Configuration類或@Component類中使用該註解,它常用於聲明一個bean,如下

@Configuration
public class Config {
    //@Bean註解用於將方法的返回值註冊為容器中的一個bean,方法返回類型就是該bean的類型,預設情況下,方法名就是bean的名稱
    //下麵這個例子:向容器中註入一個類型為ExampleA,名稱為exampleA的bean 
    @Bean
    public ExampleA exampleA() {
        return new ExampleA();
    }
}

(2) 針對@Bean方法的返回類型,也有些細節需要註意,如下所示

//有兩個介面A和B
public interface A { }

public interface B { }

//ExampleA實現了這兩個介面
public class ExampleA implements A, B { }

//而另一個bean ExampleB,它依賴了類型為B的bean
@Component
public class ExampleB {

    @Autowired
    private B b;

    public B getB() {
        return b;
    }
}

//配置類
@Configuration
@ComponentScan(basePackages = "cn.example.spring.boke")
public class Config {
    //註意,我們返回了ExampleA的實例,不過卻將返回類型聲明為了A
    @Bean
    public A exampleA() {
        return new ExampleA();
    }
}

//接著,啟動容器,會發現容器拋出NoSuchBeanDefinitionException: No qualifying bean of type 'cn.example.spring.boke.B'的異常,可見容器將返回的ExampleA僅視作了A類型然後就用於註入,雖然它也可以被視作B類型
AnnotationConfigApplicationContext ctx = new AnnotationConfigApplicationContext(Config.class);
System.out.println(ctx.getBean(ExampleB.class).getB());

//針對上面的情況,有兩種辦法解決:
//方法一:最直接,直接返回類型B而非A,如下
@Bean
public B exampleA() {
    return new ExampleA();
}

//方法二:其他的不變,只在我們的ExampleB上,加上@Lazy註解,讓容器有時間去充分的識別ExampleA,使它意識到ExampleA還可以視作B類型
@Component
@Lazy
public class ExampleB {

    //省略...
}

(3) @Bean方法可以擁有任意數量的參數,這些參數就是這個bean的依賴項,如下

@Configuration
public class Config {
    
    @Bean
    public ExampleB exampleB() {
        return new ExampleB();
    }
    
    //ExampleA依賴了ExampleB,Spring會為我們自動註入容器中的這個ExampleB對象,類似於基於構造函數的依賴註入
    @Bean
    public ExampleA exampleA(ExampleB exampleB) {
        return new ExampleA(exampleB);
    }
}

(4) 同普通的bean,由@Bean方法註入的bean也支持由JSR-250所定義的生命周期回調註解,支持InitializingBean,DisposableBean或Lifecycle等介面,支持各種Aware介面用於註入容器內置組件,同時@Bean還提供了initMethod和destroyMethod屬性用於配置初始化和銷毀回調,同<bean/>標簽的屬性,如下

public class ExampleA implements ApplicationContextAware {
    
    private ApplicationContext applicationContext;

    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        this.applicationContext = applicationContext;
    }
    
    @PostConstruct
    public void init() {
        System.out.println("init..."); 
    }
    
    public void destroy() {
        System.out.println("destroy...");
    }
}

@Configuration
public class Config {

    @Bean(destroyMethod = "destroy")
    public ExampleA exampleA() {
        return new ExampleA();
    }
}

(5) 預設情況下,當我們使用基於Java的配置來定義一個bean的時候,如果在這個bean中有聲明public修飾的close或shutdown方法,那麼這些方法會隨著該bean的銷毀回調的觸發而同時被調用,這是Spring提供的一個預設機制,無需任何的配置都會生效,但如果我們想關閉掉這一機制,可通過設置@Bean中的屬性destroyMethod=""來達到這一目的,如下

//設置destroyMethod = "",禁用掉Spring會自動觸發public修飾的close或shutdown方法的機制
@Bean(destroyMethod = "")
public ExampleA exampleA() {
    return new ExampleA();
}

(6) 我們可以通過使用@Scope註解來指定bean的作用域,如下

@Configuration
public class Config {
    //由@Bean方法所聲明的bean的作用域預設為singleton
    //@Scope註解還可用於類上
    @Bean
    @Scope("prototype")
    public ExampleA exampleA() {
        return new ExampleA();
    }
}

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • Bearer token是一種常見的身份驗證機制,通常用於Web API和其他Web服務。在前端中,Bearer token通常是通過HTTP頭(HTTP header)發送的,具體來說是通過"Authorization"頭。 在使用Bearer token進行身份驗證時,需要將token包含在HT ...
  • 在JavaScript中,可以使用 slice() 方法來截取數組的一部分。該方法接受兩個參數,第一個參數是截取的起始位置(包括該位置),第二個參數是截取的結束位置(不包括該位置)。 例如,假設有一個數組 myArray: var myArray = [1, 2, 3, 4, 5]; 要截取從第二個 ...
  • 問題 利用 map() 把字元串數組映射成整數數組 ["1", "2", "3"].map(parseInt); 是不是覺得結果應該返回 [1, 2, 3]? 但是事與願違,結果是: Output: [1, NaN, NaN] 為什麼呢? 回顧 parseInt parseInt() 函數解析一個字 ...
  • title: Hbuilder項目轉vue-cli項目 date: 2023-02-22 10:26:33 category: 工作 技能提升 cover: https://s1.ax1x.com/2022/11/23/z8L8PS.jpg tags: uniapp description: Hbu ...
  • 第一等級:代表 內聯樣式,如 style="",權值為 1,0,0,0; 第二等級:代表 ID選擇器,如 #id="", 權值為 0,1,0,0; 第三等級:代表 calss | 偽類 | 屬性 選擇器,如 .class | :hover,:link,:target | [type], 權值 0,0 ...
  • 設計意圖的傳達是架構可視化關註的重要維度,在技術方案評審過程中不可避免的會出現各種各樣的架構圖或設計圖,這些圖形化表述在設計意圖傳達效果層面表現不一,本文從圖形化的視角為軟體架構圖的評審關註點提供了參考。 ...
  • 1. 編程語言 1.1. 仍然是一門語言 1.1.1. 以最清晰、最容易理解的方式傳遞信息 1.2. 代碼的易讀性和易理解性在軟體中的重要性甚至更勝一籌 2. 領域特定語言DSL 2.1. 為瞭解決某個特定業務領域問題的一種自定義語言 2.1.1. 一種小型語言 2.1.2. 大多都不通用 2.1. ...
  • 在使用Ruoyi管理系統中出現這個問題 Error querying database. Cause: java.sql.SQLSyntaxErrorException: Unknown column 'xxx_time' in 'where clause' 因為對應報錯的SQL中沒有該欄位,前端也 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...