spring-"lookup-method"

来源:互联网 发布:证大财富淘宝贷容易吗 编辑:程序博客网 时间:2024/06/05 20:28

spring-"lookup-method"  

2009-10-23 21:14:51|  分类: java|举报|字号 订阅

下载LOFTER客
单例类调用原型类,则被调用的原形类仍是单例模式(即单例模式覆盖了原型模式),通过lookup-method属性给单例类注入原型模式类的不同的实例。
例题:
控制器StudentAction
业务类StudentService,save方法,返回值为boolean类型
想要实现单例类StudentAction调用StudentService的不同的实例
在applicationContext.xml中完成注册
<bean id="/student" class="控制器的全名">
   <!-- 通过lookup-method属性给单例类注入原型模式类的不同的实例-->
   <lookup-method bean="原型模式类的注册名" name="getter方法的名字">
</bean>
在StudentAction中调用业务层方法时,要
通过getter方法在单例类中获取原型模式类的不同的实例
boolean flag=this.getStudentService().save(参数)



方法注入不同于set注入和构造注入. 
主要是使用场景不同:一个singleton的Bean需要引用一个prototype的Bean; 一个无状态的Bean需要引用一个有状态的Bean; ... ; 等等情景下. 

试想,我们的容器创建之后就加载了所有的Bean. 而BeanA中需要引用的BeanB是状态个不确定的Bean. 那么我们需要在每次需要BeanB的时候都重新让容器加载一次吗? 真是个笨拙的想法. 

好在我们有cglib. 
cglib为我们动态的构造BeanB的子类, 当我们的BeanA需要BeanB的时候, cglib把这个子类对象给BeanA. 

看代码说话:

Java代码 
1. 
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. 
6.public abstract class HpServiceImpl implements HpService   
7.{   
8.    // protected修饰   
9.    protected HpDao hpDao = getHpDao();   
10.       
11.     
15.    public abstract HpDao getHpDao();   
16.       
17.    @Override 
18.    public int getHp()   
19.    {   
20.        // 调用原型BeanB的方法   
21.        return hpDao.getHp();   
22.    }   
23.} 

再看看BeanA的定义:

 

Xml代码            
<bean id="hpService" class="org.mycompany.spring.aop.service.impl.HpServiceImpl">
<lookup-method name="getHpDao" bean="hpDao" />
</bean>   


 

解释: 
lookup-method中name是BeanA中的抽象方法的名字.这里这个方法的用途是得到由cglib构造的BeanB的动态子类.

lookup-mentod中bean是BeanB的类型.也就是cglib构造的BeanB的动态子类的父类型...目的当然是父类的句柄可以引用子类:用这个父类型引用构造出来的动态子类.


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

 

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

讲解完毕. 
实验一下:

Java代码 

public static void main(String[] args)
{
   ApplicationContext context = new ClassPathXmlApplicationContext("classpath*:spring/application-*.xml");
   // print: false
   if(context.getBean("hpDao") == context.getBean("hpDao"))
   {
    System.out.println(true);
 

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

额外注意一下: 
如果BeanB没有写:scope="prototype" ... 
那么会出异常并提示你说: 
No Scope registered for scope 'Prototype' 
如果你添加scope='Prototype'...那还见鬼咧.还是这个错. 
所以我说了嘛...一定是小写的prototype 

   }else
   {
    System.out.println(false);
   }
  
   // print: true
   if(context.getBean("hpService") == context.getBean("hpService"))
   {
    System.out.println(true);
  
   }else
   {
    System.out.println(false);
   }
}

0 0
原创粉丝点击