Spring的方法注入lookup-method

本文介绍Spring框架中的方法注入,一种不同于构造注入和属性注入的依赖注入方式,适用于Singleton Bean引用Prototype Bean等特定场景。通过实例演示如何利用cglib动态创建Bean实例。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

方法注入不同于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出的这个异常太蛊祸人了. 在这里我提出批评. 
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值