首先要分辨BeanFactory 與 FactoryBean的區(qū)別,, 兩個名字很像,所以容易搞混
BeanFactory: 以Factory結(jié)尾,,表示它是一個工廠類,,是用于管理Bean的一個工廠
FactoryBean:以Bean結(jié)尾,表示它是一個Bean,,不同于普通Bean的是:它是實現(xiàn)了FactoryBean<T>接口的Bean,,根據(jù)該Bean的Id從BeanFactory中獲取的實際上是FactoryBean的getObject()返回的對象,而不是FactoryBean本身,, 如果要獲取FactoryBean對象,,可以在id前面加一個&符號來獲取。
spring中的Bean有兩種,。
一種是普通的bean ,,比如配置
- <bean id="personService" class="com.spring.service.impl.PersonServiceImpl" scope="prototype">
- <property name="name" value="is_zhoufeng" />
- </bean>
那個使用BeanFactory根據(jù)id personService獲取bean的時候,得到的對象就是PersonServiceImpl類型的,。
另外一種就是實現(xiàn)了org.springframework.beans.factory.FactoryBean<T>接口的Bean ,, 那么在從BeanFactory中根據(jù)定義的id獲取bean的時候,獲取的實際上是FactoryBean接口中的getObject()方法返回的對象,。
以Spring提供的ProxyFactoryBean為例子,,配置如下:
- <bean id="personServiceByLog" class="org.springframework.aop.framework.ProxyFactoryBean">
- <property name="proxyInterfaces">
- <list>
- <value>com.spring.service.PersonService</value>
- </list>
- </property>
- <property name="interceptorNames">
- <list>
- <value>logInteceptor</value>
- <value>ZFMethodAdvice</value>
- </list>
- </property>
- <property name="targetName" value="personService" />
- </bean>
那么在代碼中根據(jù)personServiceByLog來獲取的Bean實際上是PersonService類型的。
- @Test
- public void test01() {
-
- PersonService ps = context.getBean("personService", PersonService.class);
-
- ps.sayHello();
-
- String name = ps.getName();
-
- System.out.println(name);
- }
如果要獲取ProxyFactoryBean本身,,可以如下
- @Test
- public void test04() {
- ProxyFactoryBean factoryBean = context.getBean("&personServiceByLog", ProxyFactoryBean.class);
- PersonService ps = (PersonService) factoryBean.getObject();
- String name = ps.getName();
- System.out.println(name);
-
- }
自己實現(xiàn)一個FactoryBean,, 功能:用來代理一個對象,對該對象的所有方法做一個攔截,,在方法調(diào)用前后都輸出一行l(wèi)og
- package com.spring.factorybean;
-
- import java.lang.reflect.InvocationHandler;
- import java.lang.reflect.Method;
- import java.lang.reflect.Proxy;
-
- import org.springframework.beans.factory.DisposableBean;
- import org.springframework.beans.factory.FactoryBean;
- import org.springframework.beans.factory.InitializingBean;
-
- public class ZFFactoryBean implements FactoryBean<Object>, InitializingBean, DisposableBean {
-
- // 被代理對象實現(xiàn)的接口名(在使用Proxy時需要用到,,用于決定生成的代理對象類型)
- private String interfaceName;
-
- // 被代理的對象
- private Object target;
-
- // 生成的代理對象
- private Object proxyObj;
-
- public void destroy() throws Exception {
- System.out.println("distory...");
- }
-
- public void afterPropertiesSet() throws Exception {
-
- proxyObj = Proxy.newProxyInstance(this.getClass().getClassLoader(),
- new Class[] { Class.forName(interfaceName) }, new InvocationHandler() {
-
- public Object invoke(Object proxy, Method method, Object[] args)
- throws Throwable {
- System.out.println("method:" + method.getName());
- System.out.println("Method before...");
- Object result = method.invoke(target, args);
- System.out.println("Method after...");
- return result;
- }
- });
-
- System.out.println("afterPropertiesSet");
- }
-
- public Object getObject() throws Exception {
- System.out.println("getObject");
- return proxyObj;
- }
-
- public Class<?> getObjectType() {
- return proxyObj == null ? Object.class : proxyObj.getClass();
- }
-
- public boolean isSingleton() {
- return true;
- }
-
- public String getInterfaceName() {
- return interfaceName;
- }
-
- public void setInterfaceName(String interfaceName) {
- this.interfaceName = interfaceName;
- }
-
- public Object getTarget() {
- return target;
- }
-
- public void setTarget(Object target) {
- this.target = target;
- }
-
- }
然后來試試:
首先這樣定義bean
- <bean id="personService" class="com.spring.service.impl.PersonServiceImpl" scope="prototype">
- <property name="name" value="is_zhoufeng" />
- </bean>
-
- <bean id="zfPersonService" class="com.spring.factorybean.ZFFactoryBean">
- <property name="interfaceName" value="com.spring.service.PersonService" />
- <property name="target" ref="personService"/>
- </bean>
然后獲取Bean,并測試,。
- @Test
- public void test06() {
- PersonService ps = context.getBean("zfPersonService", PersonService.class);
-
- ps.sayHello();
-
- String name = ps.getName();
-
- System.out.println(name);
- }
會發(fā)現(xiàn)sayHello與getName方法調(diào)用前后都有l(wèi)og打印。
上面的ZFBeanFactory只是模仿了ProxyFactoryBean的功能做了一個實現(xiàn)而已,。
其實通過FactoryBean這種特點,,可以實現(xiàn)很多有用的功能 。,。,。
|