Spring的方法注入lookup-method

方法注入不同於set注入和構造注入. 
主要是使用場景不同:一個singleton的Bean需要引用一個prototype的Bean; 一個無狀態的Bean需要引用一個有狀態的Bean; ... ; 等等情景下. 

試想,我們的容器創建之後就加載了所有的Bean. 而BeanA中需要引用的BeanB是狀態個不確定的Bean. 那麼我們需要在每次需要BeanB的時候都重新讓容器加載一次嗎? 真是個笨拙的想法. 

好在我們有cglib. 
cglib爲我們動態的構造BeanB的子類, 當我們的BeanA需要BeanB的時候, cglib把這個子類對象給BeanA. 

看代碼說話: 
Java代碼  收藏代碼
  1. /** 
  2.  * 我們把這個類作爲BeanB.  
  3.  * (Hp就是血量... 遊戲中某個英雄的血量總在變化) 
  4.  * (比如說我們希望通過英雄名找到的英雄的血是實時變化的. ) 
  5.  */  
  6. public class HpDaoImpl implements HpDao  
  7. {  
  8.     @Override  
  9.     public int getHp()  
  10.     {  
  11.         // 返回0~100之間的隨即數  
  12.         return (int)(Math.random() * 100);  
  13.     }  
  14. }  


這是beanB的定義. 注意有個scope="prototype". 再請注意是全部小寫的prototype. 
Xml代碼  收藏代碼
  1. <bean id="hpDao" class="org.mycompany.spring.aop.dao.impl.HpDaoImpl" scope="prototype" />  


// ----------------------------------------------------------------------------- 

接下來是BeanA. 
Java代碼  收藏代碼
  1. /** 
  2.  * 我們把這個類作爲BeanA.  
  3.  * 這是一個抽象類. 爲什麼要抽象? 因爲有個抽象方法.. 
  4.  * 既然抽象了, 就不能final... 切記切記 
  5.  */  
  6. public abstract class HpServiceImpl implements HpService  
  7. {  
  8.     // protected修飾  
  9.     protected HpDao hpDao = getHpDao();  
  10.       
  11.     /* 
  12.      * 這個方法是抽象的. 
  13.      * 返回的是cglib構造的BeanB的子類. 
  14.      */  
  15.     public abstract HpDao getHpDao();  
  16.       
  17.     @Override  
  18.     public int getHp()  
  19.     {  
  20.         // 調用原型BeanB的方法  
  21.         return hpDao.getHp();  
  22.     }  
  23. }  


再看看BeanA的定義: 
Xml代碼  收藏代碼
  1. <bean id="hpService" class="org.mycompany.spring.aop.service.impl.HpServiceImpl">  
  2.     <lookup-method name="getHpDao" bean="hpDao" />  
  3. </bean>             


解釋: 
lookup-method中name是BeanA中的抽象方法的名字.這裏這個方法的用途是得到由cglib構造的BeanB的動態子類.

lookup-mentod中bean是BeanB的類型.也就是cglib構造的BeanB的動態子類的父類型...目的當然是父類的句柄可以引用子類:用這個父類型引用構造出來的動態子類. 


// ----------------------------------------------------------------------------- 

講解完畢. 
實驗一下: 
Java代碼  收藏代碼
  1. public static void main(String[] args)  
  2.     {  
  3.         ApplicationContext context = new ClassPathXmlApplicationContext("classpath*:spring/application-*.xml");  
  4.         // print: false  
  5.         if(context.getBean("hpDao") == context.getBean("hpDao"))  
  6.         {  
  7.             System.out.println(true);  
  8.           
  9.         }else  
  10.         {  
  11.             System.out.println(false);  
  12.         }  
  13.           
  14.         // print: true  
  15.         if(context.getBean("hpService") == context.getBean("hpService"))  
  16.         {  
  17.             System.out.println(true);  
  18.           
  19.         }else  
  20.         {  
  21.             System.out.println(false);  
  22.         }  
  23.     }  


// ----------------------------------------------------------------------------- 

額外注意一下: 
如果BeanB沒有寫:scope="prototype" ... 
那麼會出異常並提示你說: 
No Scope registered for scope 'Prototype
如果你添加scope='Prototype'...那還見鬼咧.還是這個錯. 
所以我說了嘛...一定是小寫的prototype 
spring出的這個異常太蠱禍人了. 在這裏我提出批評. 
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章