`
Rooock
  • 浏览: 36399 次
  • 性别: Icon_minigender_1
  • 来自: 深圳
社区版块
存档分类
最新评论

Spring的方法注入lookup-method

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

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

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

看代码说话:
/**
 * 我们把这个类作为BeanB. 
 * (Hp就是血量... 游戏中某个英雄的血量总在变化)
 * (比如说我们希望通过英雄名找到的英雄的血是实时变化的. )
 */
public class HpDaoImpl implements HpDao
{
	@Override
	public int getHp()
	{
        // 返回0~100之间的随即数
		return (int)(Math.random() * 100);
	}
}


这是beanB的定义. 注意有个scope="prototype". 再请注意是全部小写的prototype.
<bean id="hpDao" class="org.mycompany.spring.aop.dao.impl.HpDaoImpl" scope="prototype" />


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

接下来是BeanA.
/**
 * 我们把这个类作为BeanA. 
 * 这是一个抽象类. 为什么要抽象? 因为有个抽象方法..
 * 既然抽象了, 就不能final... 切记切记
 */
public abstract class HpServiceImpl implements HpService
{
    // protected修饰
	protected HpDao hpDao = getHpDao();
	
    /*
     * 这个方法是抽象的.
     * 返回的是cglib构造的BeanB的子类.
     */
	public abstract HpDao getHpDao();
	
	@Override
	public int getHp()
	{
        // 调用原型BeanB的方法
		return hpDao.getHp();
	}
}


再看看BeanA的定义:
<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的动态子类的父类型...目的当然是父类的句柄可以引用子类:用这个父类型引用构造出来的动态子类.


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

讲解完毕.
实验一下:
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);
		
		}else
		{
			System.out.println(false);
		}
		
		// print: true
		if(context.getBean("hpService") == context.getBean("hpService"))
		{
			System.out.println(true);
		
		}else
		{
			System.out.println(false);
		}
	}


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

额外注意一下:
如果BeanB没有写:scope="prototype" ...
那么会出异常并提示你说:
No Scope registered for scope 'Prototype'
如果你添加scope='Prototype'...那还见鬼咧.还是这个错.
所以我说了嘛...一定是小写的prototype
spring出的这个异常太蛊祸人了. 在这里我提出批评.
分享到:
评论
4 楼 jiazhigang 2017-07-04  
 protected HpDao hpDao = getHpDao();

确定可以这么写?编译都报错。
3 楼 four28 2016-03-11  
wanggang0321 写道
=============================================
额外注意一下:
如果BeanB没有写:scope="prototype" ...
那么会出异常并提示你说:
No Scope registered for scope 'Prototype'

=============================================
你说的这个地方,为什么我把 scope="prototype" 去掉之后,运行没有报错,而且运行成功,控制台打出来两个 true ???





是因为bean默认是singleton,所以两次获取的bean也就是一样的
2 楼 wanggang0321 2015-10-23  
=============================================
额外注意一下:
如果BeanB没有写:scope="prototype" ...
那么会出异常并提示你说:
No Scope registered for scope 'Prototype'

=============================================
你说的这个地方,为什么我把 scope="prototype" 去掉之后,运行没有报错,而且运行成功,控制台打出来两个 true ???




1 楼 chian_xxp 2010-12-28  
帅哥,能说下这样的好处不?
我叫日醒无声,你日三醒吾身。哈哈

相关推荐

Global site tag (gtag.js) - Google Analytics