天天看點

你說,怎麼把Bean塞到Spring容器?

你說,怎麼把Bean塞到Spring容器?

作者:小傅哥

部落格:https://bugstack.cn

沉澱、分享、成長,讓自己和他人都能有所收獲!😄

一、前言

小傅哥,你是怎麼學習的?

有很多初學程式設計或者碼了幾年CRUD磚的小夥伴問我,該怎麼學程式設計?感覺什麼都不會怎麼辦?感覺目前的公司沒有核心業務學到不東西呀!

其實我可能和很大一部分的粉絲讀者都有類似的經曆,在傳統類似外包的行業待過、從C#語言兩年開發再面Java崗、新到網際網路職場感覺太多不會的技術棧等等。

但可能最讓我在學習程式設計上受益的就是不斷的折騰這些技術:

  1. 關于外包:在外包2年時還是C#開發,時而搞搞中繼器、IO闆卡、PLC。但我仍舊喜歡大學時期學的Java語言,那麼每天5:30下班回家後,就不斷的用Java語言把公司接觸到的C#工程做翻新。差不多1年的時間,把幾乎我接觸到的項目翻新了個遍,就是那個時候知道的Java還能做序列槽通信,還是蠻有意思的。
  2. 關于場景:其實很多程式員在一個相對較小的公司時,學習的最大瓶頸是眼界問題,不知道有什麼技術、不知道有什麼場景,更不知道自己不會啥。其實很多時候這都跟

    有關系,公司是沒有這樣的場景,但是你可以看部落格、看論壇、看視訊,加各類技術群。如果遇到哪些發廣告的就退了,哪些好的留下,認識一些人脈,相知一些基友,這在個過程總能有所收獲,你會随着時間的推移嗅到各類技術棧、項目、經驗、心得、面試等等,當你武裝好了自己,再出去面試也就沒那麼難了。
  3. 關于開始:時間少、要學的多,感覺自己就是一把

    小鐵鍬

    ,要去挖蘇伊士運河,不知道能從哪開始。這個時候建議不要盲目的收藏幾個T的資料和視訊,先打開xmind,選個好看的主題,開始梳理自己的技術棧,看看自己會什麼不會什麼,在從這些不會的内容裡選出你最想學的,把要學的内容在梳理出相應的資料庫。好,那麼這個時候你就可以開始了,記住開始是從一點點深入的,不要總想着一口吃個胖子。

方向對了,快是最大的障礙!

,很多時候隻要你能平心靜氣日積月累的學習,其實就沒有什麼不能克服的問題。程式設計裡又有什麼非常難的東西嗎,大部分知識都是不知道就不會而已,知道了就很簡單。

二、面試題

謝飛機,小記!

,履歷上我都寫精通了,要個20K沒問題,等着吧!

面試官:謝飛機,技術不錯呀,都是精通,哦,有一個vb了解,沒事我們不用vb

謝飛機:還行,我學的多,你問吧。

面試官:嗯,自信了不少。那我們聊聊 Spring,你這個也寫的精通。

謝飛機:來吧!

面試官:你說,怎麼把Bean塞到Spring容器?能說說它的過程嗎,你有過相關技術的使用嗎,應用了什麼場景?

謝飛機:嗯!?嗯,,好像,沒用過。我都是精通使用API,@Resource

面試官:哦,@Resource,注解是Spring哪個子產品提供的?

謝飛機:我,,,再見!ヾ( ̄▽ ̄)

三、代理Bean注冊到Spring容器

你說,怎麼把Bean塞到Spring容器?
  • 關于Bean注冊的技術場景,在我們日常用到的技術架構中,MyBatis 是最為常見的。通過在使用 MyBatis 時都隻是定義一個接口不需要寫實作類,但是這個接口卻可以和配置的 SQL 語句關聯,執行相應的資料庫操作時可以傳回對應的結果。那麼這個接口與資料庫的操作就用到的 Bean 的代理和注冊。
  • 我們都知道類的調用是不能直接調用沒有實作的接口的,是以需要通過代理的方式給接口生成對應的實作類。接下來再通過把代理類放到 Spring 的 FactoryBean 的實作中,最後再把這個 FactoryBean 實作類注冊到 Spring 容器。那麼現在你的代理類就已經被注冊到 Spring 容器了,接下來就可以通過注解的方式注入到屬性中。

按照這個實作方式,我們來操作一下,看看一個 Bean 的注冊過程在代碼中是如何實作的。

1. 定義接口

public interface IUserDao {

    String queryUserInfo();

}
           
  • 先定義一個類似 DAO 的接口,基本這樣的接口在使用 MyBatis 時還是非常常見的。後面我們會對這個接口做代理和注冊。

2. 類代理實作

ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
Class<?>[] classes = {IUserDao.class};    

InvocationHandler handler = (proxy, method, args) -> "你被代理了 " + method.getName();
IUserDao userDao = (IUserDao) Proxy.newProxyInstance(classLoader, classes, handler); 

String res = userDao.queryUserInfo();
logger.info("測試結果:{}", res);
           
  • Java 本身的代理方式使用起來還是比較簡單的,用法也很固定。
  • InvocationHandler 是個接口類,它對應的實作内容就是代理對象的具體實作。
  • 最後就是把代理交給 Proxy 建立代理對象,

    Proxy.newProxyInstance

3. 實作Bean工廠

public class ProxyBeanFactory implements FactoryBean {

    @Override
    public Object getObject() throws Exception {

        ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
        Class[] classes = {IUserDao.class};
        InvocationHandler handler = (proxy, method, args) -> "你被代理了 " + method.getName();

        return Proxy.newProxyInstance(classLoader, classes, handler);
    }

    @Override
    public Class<?> getObjectType() {
        return IUserDao.class;
    } 

}
           
  • FactoryBean 在 spring 起到着二當家的地位,它将近有70多個小弟(實作它的接口定義),那麼它有三個方法;
    • T getObject() throws Exception; 傳回bean執行個體對象
    • Class<?> getObjectType(); 傳回執行個體類類型
    • boolean isSingleton(); 判斷是否單例,單例會放到Spring容器中單執行個體緩存池中
  • 在這裡我們把上面使用Java代理的對象放到了 getObject() 方法中,那麼現在再從 Spring 中擷取到的對象,就是我們的代理對象了。

4. Bean 注冊

public class RegisterBeanFactory implements BeanDefinitionRegistryPostProcessor {

    @Override
    public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException {

        GenericBeanDefinition beanDefinition = new GenericBeanDefinition();
        beanDefinition.setBeanClass(ProxyBeanFactory.class);

        BeanDefinitionHolder definitionHolder = new BeanDefinitionHolder(beanDefinition, "userDao");
        BeanDefinitionReaderUtils.registerBeanDefinition(definitionHolder, registry);
    }

}
           

在 Spring 的 Bean 管理中,所有的 Bean 最終都會被注冊到類 DefaultListableBeanFactory 中,以上這部分代碼主要的内容包括:

  • 實作 BeanDefinitionRegistryPostProcessor.postProcessBeanDefinitionRegistry方法,擷取 Bean 注冊對象。
  • 定義 Bean,GenericBeanDefinition,這裡主要設定了我們的代理類工廠。
  • 建立 Bean 定義處理類,BeanDefinitionHolder,這裡需要的主要參數;定義 Bean 和名稱

    setBeanClass(ProxyBeanFactory.class)

  • 最後将我們自己的bean注冊到spring容器中去,registry.registerBeanDefinition()

四、測試驗證

在上面我們已經把自定義代理的 Bean 注冊到了 Spring 容器中,接下來我們來測試下這個代理的 Bean 被如何調用。

1. 定義 spring-config.xml

<bean id="userDao" class="org.itstack.interview.bean.RegisterBeanFactory"/>
           
  • 這裡我們把 RegisterBeanFactory 配置到 spring 的 xml 配置中,便于啟動時加載。

2. 單元測試

@Test
public void test_IUserDao() {
    BeanFactory beanFactory = new ClassPathXmlApplicationContext("spring-config.xml");
    IUserDao userDao = beanFactory.getBean("userDao", IUserDao.class);
    String res = userDao.queryUserInfo();
    logger.info("測試結果:{}", res);
}
           

測試結果

22:53:14.759 [main] DEBUG o.s.c.e.PropertySourcesPropertyResolver - Could not find key 'spring.liveBeansView.mbeanDomain' in any property source
22:53:14.760 [main] DEBUG o.s.b.f.s.DefaultListableBeanFactory - Returning cached instance of singleton bean 'userDao'
22:53:14.796 [main] INFO  org.itstack.interview.test.ApiTest - 測試結果:你被代理了 queryUserInfo

Process finished with exit code 0
           
  • 從測試結果可以看到,我們已經可以通過注入到Spring的代理Bean對象,實作我們的預期結果。
  • 其實這個過程也是很多架構中用到的方式,尤其是在一些中間件開發,類似的 ORM 架構都需要使用到。

五、總結

  • 本章節的内容相對來說非常并不複雜,隻不過這一塊的代碼是我們從源碼的學習中提取出來的最核心流程,因為在大部分架構中也基本都是這樣的進行處理的。如果這樣的地方不了解,那麼很難讀懂諸如此類的架構源碼,也很難了解它是怎麼調用的。
  • 在本文中主要涉及到的技術點包括;代理、對象、注冊,以及相應的使用。尤其是 Bean 的定義

    BeanDefinitionHolder

    和 Bean 的注冊

    BeanDefinitionReaderUtils.registerBeanDefinition

  • 如果你還能把此類技術聯想的更多,可以嘗試把代理的對象替換成資料庫的查詢對象,也就是對 JDBC 的操作,當你完成以後也就實作了一個簡單的 ORM 架構。其實很多技術實作都是由小做大,但最開始的那部分是整個代碼實作的核心。

六、系列推薦

  • 認知自己的技術棧盲區
  • LinkedList插入速度比ArrayList快?你确定嗎?
  • 除了JDK、CGLIB,還有3種類代理方式?面試又卡住!
  • ReentrantLock之AQS原理分析和實踐使用
  • 咋嘞?你的IDEA過期了吧!加個Jar包就破解了,為什麼?

公衆号:bugstack蟲洞棧 | 作者小傅哥多年從事一線網際網路 Java 開發的學習曆程技術彙總,旨在為大家提供一個清晰詳細的學習教程,側重點更傾向編寫Java核心内容。如果能為您提供幫助,請給予支援(關注、點贊、分享)!