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

小傅哥發表於2021-03-31


作者:小傅哥
部落格: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註冊

  • 關於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 框架。其實很多技術實現都是由小做大,但最開始的那部分是整個程式碼實現的核心。

六、系列推薦

相關文章