Elim的博客

用来记录一些原创性的总结


Elim

7 单例注入多例之lookup-method

Spring有一种机制,可以动态的实现或重写bean容器中指定bean的指定方法,然后将返回值指定为bean容器中的另一个bean。现针对前文提到的单例bean需注入多例bean的应用场景来谈谈如何使用这种机制。先假设我们有一个单例beanA需注入一个多例BeanB类型的bean,那么我们可以这样来做。

1、在beanA对应的类中定义一个方法用来获取BeanB,有这么一个方法就行,可以是空实现,到时候将由Spring来为我们重写或实现,然后返回一个BeanB类型的bean。为此,可以将我们的BeanA如下定义:

public class BeanA {

	/**
	 * 需要使用到BeanB的方法
	 */
	public void doSomething() {
		BeanB beanB = this.getBeanB();
		System.out.println(beanB);
		//...
	}
	
	/**
	 * 定义一个返回值为BeanB的方法,用来获取BeanB类型的bean,该方法将由Spring来重写。
	 * @return
	 */
	public BeanB getBeanB() {
		return null;
	}
	
}

2、在ApplicationContext中定义一个单例beanA和一个多例beanB,同时通过lookup-method元素指定beanAgetBeanB方法将会被Spring重写并返回bean容器中的beanB

	<bean id="beanB" class="com.app.BeanB" scope="prototype"/>
	<bean id="beanA" class="com.app.BeanA">
		<!-- 表示将由Spring重写getBeanB()方法,并返回名为beanB的bean -->
		<lookup-method name="getBeanB" bean="beanB"/>
	</bean>

经过以上两步以后每次在调度beanAgetBeanB()方法时,Spring都重新从bean容器中获取一个beanB,因为beanB定义为多例形式,所以每次都会获取一个全新的BeanB对象。

在给一个bean指定了lookup-method后,Spring将通过CGLIB动态的生成一个该bean对应类型的子类,然后在该子类中实现或重写lookup-method元素指定的方法,并从bean容器中获取lookup-method元素指定的bean作为返回值进行返回。当bean指定的类型是一个抽象类且lookup-method指定的方法是一个抽象方法时,Spring就将实现该抽象方法,否则就是重写。定义了lookup-methodbean真正的类型是Spring动态生成类的类型,但是它可以被当做bean本身指定的类型使用,因为动态生成的类就是继承自bean本身指定的类型的。

由于Spring是需要动态生成类来重写或实现指定的方法的,所以我们必须确保由lookup-method指定的方法是可以被重写的,这就要求该方法对子类是可访问的,而且不能是final型的。具体来讲lookup-method元素指定的方法需要具有如下形式:

<public|protected> [abstract] <return-type> methodName(no-arguments)

如你所见,lookup-method指定的方法还不能有参数。

(注:本文是基于Spring4.1.0所写)