So after much hair-pulling and thought, I was able to extract a few interfaces and configure some of the business logic classes as Spring managed beans. The problem was that some of the dependencies were still being acquired via the "new" keyword, and it had to remain that way.
In other words, I had a singleton bean which required a non-singleton, or prototype. Not just once, at bean creation time, but a new instance every time. How could this be done, without mucking up my code with the Spring API? We definitely do not want a reference to a BeanFactory or ApplicationContext in our bean.
Enter one of the rarer forms of dependency injection: Method Injection (see section 3.3.8).
The recipe is:
- Create a method for the dependency and either make it protected abstract, or public.
- In your applicationContext.xml, instead of specifying <property name="someProperty">, you use
<lookup-method name="theMethodWhichReturnsYourPrototype" bean="idOfPrototypeBean"/>
Let's look at an example:
We have a class Homer, which wants a new instance of another bean, a Donut, every time, instead of reusing the same Donut over and over again.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
package demo; | |
public abstract class Homer implements EatsDonuts{ | |
public Homer() { } | |
public void eat() { | |
Donut d = getDonut(); //We want a new donut every time | |
//eat donut | |
} | |
//here is where Spring will do its magic and return a prototype | |
protected abstract Donut getDonut(); | |
// I actually prefer to do it this way, so that we don't have to make the class | |
// abstract: | |
/* | |
public Donut getDonut() { | |
return null; //if we get NPE, we know something went wrong in our xml file | |
// or we can have: | |
// return new TastyDonut(); // so that the legacy code stays the same | |
} | |
*/ | |
} |
Now, for the applicationContext.xml:
<!-- a stateful bean deployed as a prototype (non-singleton) --> <bean id="donut" class="demo.DonutImpl" scope="prototype"> <!-- inject dependencies here as required --> </bean> <!-- homer (singleton) uses donut (prototype) --> <bean id="homer" class="demo.Homer"> <lookup-method name="getDonut" bean="donut"/> </bean>
So what's going on here? To quote the Spring docs:
The Spring Framework implements this method injection by dynamically generating a subclass overriding the method, using bytecode generation via the CGLIB library.
So when Homer asks for a donut via getDonut(), the subclassed Homer, generated via CGLIB, will return a new instance of Donut, which has its dependencies injected by Spring.
Pros of this approach: we are decoupled from BeanFactory and the Spring API
Cons: making the bean abstract makes testing a little difficult, since you have to subclass the class, however, making it concrete and implementing the getter method, you avoid this.
No comments:
Post a Comment