文章目录
- 一、目标:对象作用域和FactoryBean
 - 二、设计:对象作用域和FactoryBean
 - 三、实现:对象作用域和FactoryBean
 - 3.0 引入依赖
 - 3.1 工程结构
 - 3.2 Spring单例、原型以及FactoryBean功能类图
 - 3.3 Bean的作用范围定义和xml解析
 - 3.3.1 Bean对象信息定义
 - 3.3.2 XML处理Bean注册
 - 3.3.3 创建和修改对象时判断单例和原型模式
 
- 3.4 定义FactoryBean接口和注册服务
 - 3.4.1 定义FactoryBean接口
 - 3.4.2 通用的注册表实现
 - 3.4.3 实现 FactoryBean 注册服务
 - 3.4.4 扩展 AbstractBeanFactory创建对象逻辑
 
- 四、测试:对象作用域和FactoryBean
 - 4.1 添加测试配置
 - 4.1.1 添加IUserDao接口
 - 4.1.2 修改UserService用户对象
 - 4.1.3 定义FactoryBean对象
 - 4.1.4 配置文件
 
- 4.2 单元测试
 - 4.2.1 单例&原型测试
 - 4.2.2 代理对象测试
 
- 五、总结:对象作用域和FactoryBean
 
一、目标:对象作用域和FactoryBean
💡 交给 Spring 管理的 Bean 对象,一定就是我们用类创建出来的 Bean 吗?创建出来的 Bean 就永远是单例吗,没有可能是原型模式吗?
- 在 Spring 框架中,使用 Mybatis 框架,它的核心作用是可以满足用户不需要实现 Dao 接口类,就可以通过 xml 或者注解配置的方式完成对数据库执行 CRUD 操作,那么在实现这样的 ORM 框架中,是怎么把一个数据操作的 Bean 对象交给 Spring 管理的。
 - 我们在使用 Spring、Mybatis 框架的时候知道,并没有手动的去创建任何操作数据库的 Bean 对象,有的仅仅是一个接口定义,而这个接口定义竟然可以被注入到其他需要使用 Dao 的属性中去。 
  
- 这一过程最核心待解决的问题:就是需要完成把复杂且以代理方式动态变化的对象,注册到 Spring 容器中。
 
 
二、设计:对象作用域和FactoryBean
💡 设计:提供一个能让使用者定义复杂的 Bean 对象。
- 提供一个能让使用者定义复杂的 Bean 对象,这样的功能逻辑设计上并不复杂。因为 Spring 框架在开发的过程中就已经提供了各项扩展能力的 
接口。- 你只需要在合适的位置提供一个 
接口的处理接口调用和相应的功能逻辑实现即可。 - 如:对外提供一个可以二次从 
FactoryBean的getObject方法中获取对象的功能即可。这样所有实现此接口的对象类,就可以扩充自己的对象功能了。 
 - 你只需要在合适的位置提供一个 
 - Mybatis 就是实现了一个 
MapperFactoryBean类,在getObject方法中提供SqlSession对执行 CRUD 方法的操作。 

- 整个实现过程包括两部分,一个解决单例还是原型对象,另外一个处理 
FactoryBean类型对象创建过程中关于获取具体调用对象的getObject操作。 SCOPE_SINGLETON、SCOPE_PROTOTYPE,对象类型的创建获取方式,主要区分在于:AbstractAutowireCapableBeanFactory#createBean创建完成对象后是否放入内存中,如果不放入则每次获取都会重新创建。
createBean执行对象创建、属性填充、依赖加载、前置后置处理、初始化等操作后。就要开始做执行判断整个对象是否是一个FactoryBean对象。- 如果是这样的对象,就需要再继续执行获取 
FactoryBean具体对象中的getObject对象了。 - 整个 
getBean过程中都会新增一个单例类型的判断factory.isSingleton(),用于决定是否使用内存存放对象信息。 
- 如果是这样的对象,就需要再继续执行获取 
 
三、实现:对象作用域和FactoryBean
3.0 引入依赖
<!-- https://mvnrepository.com/artifact/org.openjdk.jol/jol-cli -->
<dependency>
    <groupId>org.openjdk.jol</groupId>
    <artifactId>jol-cli</artifactId>
    <version>0.14</version>
</dependency>
 
3.1 工程结构
spring-step-09
|-src
	|-main
	|	|-java
	|		|-com.lino.springframework
	|			|-beans
	|			|	|-factory
	|			|	|	|-config
	|			|	|	|	|-AutowireCapableBeanFactory.java
	|			|	|	|	|-BeanDefinition.java
	|			|	|	|	|-BeanFactoryPostProcessor.java
	|			|	|	|	|-BeanPostProcessor.java
	|			|	|	|	|-BeanReference.java
	|			|	|	|	|-ConfigurableBeanFactory.java
	|			|	|	|	|-SingletonBeanRegistry.java
	|			|	|	|-support
	|			|	|	|	|-AbstractAutowireCapableBeanFactory.java
	|			|	|	|	|-AbstractBeabDefinitionReader.java
	|			|	|	|	|-AbstractBeabFactory.java
	|			|	|	|	|-BeabDefinitionReader.java
	|			|	|	|	|-BeanDefinitionRegistry.java
	|			|	|	|	|-CglibSubclassingInstantiationStrategy.java
	|			|	|	|	|-DefaultListableBeanFactory.java
	|			|	|	|	|-DefaultSingletonBeanRegistry.java
	|			|	|	|	|-DisposableBeanAdapter.java
	|			|	|	|	|-FactoryBeanRegistrySupport.java
	|			|	|	|	|-InstantiationStrategy.java
	|			|	|	|	|-SimpleInstantiationStrategy.java
	|			|	|	|-support
	|			|	|	|	|-XMLBeanDefinitionReader.java
	|			|	|	|-Aware.java
	|			|	|	|-BeanClassLoaderAware.java
	|			|	|	|-BeanFactory.java
	|			|	|	|-BeanFactoryAware.java
	|			|	|	|-BeanNameAware.java
	|			|	|	|-ConfigurableListableBeanFactory.java
	|			|	|	|-DisposableBean.java
	|			|	|	|-FactoryBean.java
	|			|	|	|-HierarcgicalBeanFactory.java
	|			|	|	|-InitializingBean.java
	|			|	|	|-ListableBeanFactory.java
	|			|	|-BeansException.java
	|			|	|-PropertyValue.java
	|			|	|-PropertyValues.java
	|			|-context
	|			|	|-support
	|			|	|	|-AbstractApplicationContext.java
	|			|	|	|-AbstractRefreshableApplicationContext.java
	|			|	|	|-AbstractXmlApplicationContext.java
	|			|	|	|-ApplicationContextAwareProcessor.java
	|			|	|	|-ClassPathXmlApplicationContext.java
	|			|	|-ApplicationContext.java
	|			|	|-ApplicationContextAware.java
	|			|	|-ConfigurableApplicationContext.java
	|			|-core.io
	|			|	|-ClassPathResource.java
	|			|	|-DefaultResourceLoader.java
	|			|	|-FileSystemResource.java
	|			|	|-Resource.java
	|			|	|-ResourceLoader.java
	|			|	|-UrlResource.java
	|			|-util
	|			|	|-ClassUtils.java
	|-test
		|-java
			|-com.lino.springframework.test
                |-bean
                |	|-IUserDao.java
                |	|-ProxyBeanFactory.java
                |	|-UserService.java
                |-ApiTest.java
		|-resources
			|-spring.xml
 
3.2 Spring单例、原型以及FactoryBean功能类图

- 整个类图展示的就是添加 Bean 的实例化是单例还是原型模式以及 FactoryBean 的实现。
 - 实现:只是在现有的 
AbstractAutowireCapableBeanFactory类以及继承的抽象类AbstractBeanFactory中进行扩展。 - 不过这次要把 
AbstractBeanFactory继承的DefaultSingletonBeanRegistry类,中间加一层FactoryBeanRegistrySupport,这个类在 Spring 框架中主要是处理关于 FactoryBean 注册的支撑操作。 
3.3 Bean的作用范围定义和xml解析
3.3.1 Bean对象信息定义
BeanDefinition.java
package com.lino.springframework.beans.factory.config;
import com.lino.springframework.beans.PropertyValues;
/**
 * @description: Bean 对象信息定义
 */
public class BeanDefinition {
    String SCOPE_SINGLETON = ConfigurableBeanFactory.SCOPE_SINGLETON;
    String SCOPE_PROTOTYPE = ConfigurableBeanFactory.SCOPE_PROTOTYPE;
    private Class beanClass;
    private PropertyValues propertyValues;
    private String initMethodName;
    private String destroyMethodName;
    private String scope = SCOPE_SINGLETON;
    private boolean singleton = true;
    private boolean prototype = false;
    public BeanDefinition(Class beanClass) {
        this(beanClass, null);
    }
    public BeanDefinition(Class beanClass, PropertyValues propertyValues) {
        this.beanClass = beanClass;
        this.propertyValues = propertyValues != null ? propertyValues : new PropertyValues();
    }
    public void setScope(String scope) {
        this.scope = scope;
        this.singleton = SCOPE_SINGLETON.equals(scope);
        this.prototype = SCOPE_PROTOTYPE.equals(scope);
    }
    public boolean isSingleton() {
        return singleton;
    }
    public boolean isPrototype() {
        return prototype;
    }
    public Class getBeanClass() {
        return beanClass;
    }
    public void setBeanClass(Class beanClass) {
        this.beanClass = beanClass;
    }
    public PropertyValues getPropertyValues() {
        return propertyValues;
    }
    public void setPropertyValues(PropertyValues propertyValues) {
        this.propertyValues = propertyValues;
    }
    public String getInitMethodName() {
        return initMethodName;
    }
    public void setInitMethodName(String initMethodName) {
        this.initMethodName = initMethodName;
    }
    public String getDestroyMethodName() {
        return destroyMethodName;
    }
    public void setDestroyMethodName(String destroyMethodName) {
        this.destroyMethodName = destroyMethodName;
    }
}
 
singleton、prototype,是本次在 BeanDefinition 类中新增加的两个属性信息。- 用于把从 
spring.xml中解析到的 Bean 对象作用范围填充到属性中。 
- 用于把从 
 
3.3.2 XML处理Bean注册
XmlBeanDefinitionReader.java
package com.lino.springframework.beans.factory.xml;
import cn.hutool.core.util.StrUtil;
import cn.hutool.core.util.XmlUtil;
import com.lino.springframework.beans.BeansException;
import com.lino.springframework.beans.PropertyValue;
import com.lino.springframework.beans.factory.config.BeanDefinition;
import com.lino.springframework.beans.factory.config.BeanReference;
import com.lino.springframework.beans.factory.support.AbstractBeanDefinitionReader;
import com.lino.springframework.beans.factory.support.BeanDefinitionRegistry;
import com.lino.springframework.core.io.Resource;
import com.lino.springframework.core.io.ResourceLoader;
import org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.w3c.dom.NodeList;
import java.io.IOException;
import java.io.InputStream;
/**
 * @description: XML处理Bean注册
 */
public class XmlBeanDefinitionReader extends AbstractBeanDefinitionReader {
    ...
    protected void doLoadBeanDefinitions(InputStream inputStream) throws ClassNotFoundException {
        Document doc = XmlUtil.readXML(inputStream);
        Element root = doc.getDocumentElement();
        NodeList childNodes = root.getChildNodes();
        for (int i = 0; i < childNodes.getLength(); i++) {
            // 判断元素
            if (!(childNodes.item(i) instanceof Element)) {
                continue;
            }
            // 判断对象
            if (!"bean".equals(childNodes.item(i).getNodeName())) {
                continue;
            }
            // 解析标签
            Element bean = (Element) childNodes.item(i);
            String id = bean.getAttribute("id");
            String name = bean.getAttribute("name");
            String className = bean.getAttribute("class");
            String initMethod = bean.getAttribute("init-method");
            String destroyMethodName = bean.getAttribute("destroy-method");
            String beanScope = bean.getAttribute("scope");
            // 获取 Class, 方便获取类中的名称
            Class<?> clazz = Class.forName(className);
            // 优先级 id > name
            String beanName = StrUtil.isNotEmpty(id) ? id : name;
            if (StrUtil.isEmpty(beanName)) {
                beanName = StrUtil.lowerFirst(clazz.getSimpleName());
            }
            // 定义bean
            BeanDefinition beanDefinition = new BeanDefinition(clazz);
            beanDefinition.setInitMethodName(initMethod);
            beanDefinition.setDestroyMethodName(destroyMethodName);
            if (StrUtil.isNotEmpty(beanScope)) {
                beanDefinition.setScope(beanScope);
            }
            // 读取属性并填充
            for (int j = 0; j < bean.getChildNodes().getLength(); j++) {
                // 判断元素
                if (!(bean.getChildNodes().item(j) instanceof Element)) {
                    continue;
                }
                // 判断对象
                if (!"property".equals(bean.getChildNodes().item(j).getNodeName())) {
                    continue;
                }
                // 解析标签:property
                Element property = (Element) bean.getChildNodes().item(j);
                String attrName = property.getAttribute("name");
                String attrValue = property.getAttribute("value");
                String attrRef = property.getAttribute("ref");
                // 获取属性值:引入对象、值对象
                Object value = StrUtil.isNotEmpty(attrRef) ? new BeanReference(attrRef) : attrValue;
                // 创建属性信息
                PropertyValue propertyValue = new PropertyValue(attrName, value);
                beanDefinition.getPropertyValues().addPropertyValue(propertyValue);
            }
            if (getRegistry().containsBeanDefinition(beanName)) {
                throw new BeansException("Duplicate beanName[" + beanName + "] is not allowed");
            }
            // 注册 BeanDefinition
            getRegistry().registerBeanDefinition(beanName, beanDefinition);
        }
    }
}
 
- 在解析 XML 处理类 XmlBeanDefinitionReader 中,新增加了关于 Bean 对象配置中 
scope的解析,并把这个属性信息填充到 Bean 定义中。beanDefinition.setScope(beanScope)
 
3.3.3 创建和修改对象时判断单例和原型模式
AbstractAutowireCapableBeanFactory.java
package com.lino.springframework.beans.factory.support;
import cn.hutool.core.bean.BeanUtil;
import cn.hutool.core.util.StrUtil;
import com.lino.springframework.beans.BeansException;
import com.lino.springframework.beans.PropertyValue;
import com.lino.springframework.beans.PropertyValues;
import com.lino.springframework.beans.factory.*;
import com.lino.springframework.beans.factory.config.AutowireCapableBeanFactory;
import com.lino.springframework.beans.factory.config.BeanDefinition;
import com.lino.springframework.beans.factory.config.BeanPostProcessor;
import com.lino.springframework.beans.factory.config.BeanReference;
import java.lang.reflect.Constructor;
import java.lang.reflect.Method;
/**
 * @description: 实现默认bean创建的抽象bean工厂超类
 */
public abstract class AbstractAutowireCapableBeanFactory extends AbstractBeanFactory implements AutowireCapableBeanFactory {
    private InstantiationStrategy instantiationStrategy = new CglibSubclassingInstantiationStrategy();
    @Override
    protected Object createBean(String beanName, BeanDefinition beanDefinition, Object[] args) {
        Object bean = null;
        try {
            bean = createBeanInstance(beanDefinition, beanName, args);
            // 给bean填充属性
            applyPropertyValues(beanName, bean, beanDefinition);
            // 执行 Bean 的初始化方法和 BeanPostProcessor 的前置和后置处理方法
            bean = initializeBean(beanName, bean, beanDefinition);
        } catch (Exception e) {
            throw new BeansException("Instantiation of bean failed", e);
        }
        // 注册实现 DisposableBean 接口的 Bean 对象
        registerDisposableBeanIfNecessary(beanName, bean, beanDefinition);
        // 判断 SCOPE_SINGLETON、SCOPE_PROTOTYPE
        if (beanDefinition.isSingleton()) {
            registerSingletonBean(beanName, bean);
        }
        return bean;
    }
    private void registerDisposableBeanIfNecessary(String beanName, Object bean, BeanDefinition beanDefinition) {
        // 非 Singleton 类型的 Bean 不执行销毁方法
        if (!beanDefinition.isSingleton()) {
            return;
        }
        if (bean instanceof DisposableBean || StrUtil.isNotEmpty(beanDefinition.getDestroyMethodName())) {
            registerDisposableBean(beanName, new DisposableBeanAdapter(bean, beanName, beanDefinition));
        }
    }
    ...
}
 
- 单例模式和原型模式的区别就在于是否存放到内存中。 
  
- 如果是原型模式那么就不会存放到内存中,每次获取都重新创建对象。
 - 另外非 Singleton 类型的 Bean 不需要执行销毁方法。
 
 
3.4 定义FactoryBean接口和注册服务
3.4.1 定义FactoryBean接口
FactoryBean.java
package com.lino.springframework.beans.factory;
/**
 * @description: 工厂对象接口
 */
public interface FactoryBean<T> {
    /**
     * 获取对象
     *
     * @return 泛型对象
     * @throws Exception 异常
     */
    T getObject() throws Exception;
    /**
     * 获取对象类型
     *
     * @return 对象类型
     */
    Class<?> getObjectType();
    /**
     * 是否单例对象
     *
     * @return 是否:布尔值
     */
    boolean isSingleton();
}
 
- FactoryBean 中需要提供3个方法,获取对象、对象类型、以及是否是单例对象,如果是单例对象依然会被放入内存中。
 
3.4.2 通用的注册表实现
DefaultSingletonBeanRegistry.java
package com.lino.springframework.beans.factory.support;
import com.lino.springframework.beans.BeansException;
import com.lino.springframework.beans.factory.DisposableBean;
import com.lino.springframework.beans.factory.config.SingletonBeanRegistry;
import java.util.HashMap;
import java.util.Map;
import java.util.Set;
/**
 * @description: 通用的注册表实现
 */
public class DefaultSingletonBeanRegistry implements SingletonBeanRegistry {
    protected static final Object NULL_OBJECT = new Object();
    private Map<String, Object> singletonObjects = new HashMap<>();
    private final Map<String, DisposableBean> disposableBeans = new HashMap<>();
    ...
}
 
- 添加空对象 
Object NULL_OBJECT = new Object()。 
3.4.3 实现 FactoryBean 注册服务
FactoryBeanRegistrySupport.java
package com.lino.springframework.beans.factory.support;
import com.lino.springframework.beans.BeansException;
import com.lino.springframework.beans.factory.FactoryBean;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
/**
 * @description: FactoryBean注册对象
 */
public abstract class FactoryBeanRegistrySupport extends DefaultSingletonBeanRegistry {
    /**
     * FactoryBean对象缓存
     */
    private final Map<String, Object> factoryBeanObjectCache = new ConcurrentHashMap<>();
    protected Object getCachedObjectForFactoryBean(String beanName) {
        Object object = this.factoryBeanObjectCache.get(beanName);
        return (object != NULL_OBJECT ? object : null);
    }
    protected Object getObjectFromFactoryBean(FactoryBean factory, String beanName) {
        if (factory.isSingleton()) {
            Object object = this.factoryBeanObjectCache.get(beanName);
            if (object == null) {
                object = doGetObjectFromFactoryBean(factory, beanName);
                this.factoryBeanObjectCache.put(beanName, (object != null ? object : NULL_OBJECT));
            }
            return (object != NULL_OBJECT ? object : null);
        } else {
            return doGetObjectFromFactoryBean(factory, beanName);
        }
    }
    private Object doGetObjectFromFactoryBean(final FactoryBean factory, final String beanName) {
        try {
            return factory.getObject();
        } catch (Exception e) {
            throw new BeansException("FactoryBean threw exception on object[" + beanName + "] creation", e);
        }
    }
}
 
- FactoryBeanRegistrySupport 类主要处理的就是关于 FactoryBean 此类对象的注册操作,之所以放到这样一个单独的类里,就是希望做到不同领域模块下只负责各自需要完成的功能,避免因为扩展导致类膨胀到难以维护。
 - 同时这里也定义了缓存操作 
factoryBeanObjectCache,用于存放单例类型的对象,避免重复创建。- 日常使用中,基本都是创建的单例对象。
 
 doGetObjectFromFactoryBean是具体的获取FactoryBean#getObject()方法。- 因为既有缓存的处理也有对象的获取,所以额外提供了 
getObjectFromFactoryBean进行逻辑包装。 
- 因为既有缓存的处理也有对象的获取,所以额外提供了 
 
3.4.4 扩展 AbstractBeanFactory创建对象逻辑
AbstractBeanFactory.java
package com.lino.springframework.beans.factory.support;
import com.lino.springframework.beans.BeansException;
import com.lino.springframework.beans.factory.FactoryBean;
import com.lino.springframework.beans.factory.config.BeanDefinition;
import com.lino.springframework.beans.factory.config.BeanPostProcessor;
import com.lino.springframework.beans.factory.config.ConfigurableBeanFactory;
import com.lino.springframework.util.ClassUtils;
import java.util.ArrayList;
import java.util.List;
/**
 * @description: 抽象的 Bean 工厂基类,定义模板方法
 */
public abstract class AbstractBeanFactory extends FactoryBeanRegistrySupport implements ConfigurableBeanFactory {
    private ClassLoader beanClassLoader = ClassUtils.getDefaultClassLoader();
    private final List<BeanPostProcessor> beanPostProcessors = new ArrayList<>();
    ...
    protected <T> T doGetBean(final String name, final Object[] args) {
        Object sharedInstance = getSingleton(name);
        if (null != sharedInstance) {
            // 如果是 FactoryBean,则需要调用 FactoryBean#getObject
            return (T) getObjectFromBeanInstance(sharedInstance, name);
        }
        BeanDefinition beanDefinition = getBeanDefinition(name);
        Object bean = createBean(name, beanDefinition, args);
        return (T) getObjectFromBeanInstance(bean, name);
    }
    private Object getObjectFromBeanInstance(Object beanInstance, String beanName) {
        if (!(beanInstance instanceof FactoryBean)) {
            return beanInstance;
        }
        Object object = getCachedObjectForFactoryBean(beanName);
        if (object == null) {
            FactoryBean<?> factoryBean = (FactoryBean<?>) beanInstance;
            object = getObjectFromFactoryBean(factoryBean, beanName);
        }
        return object;
    }
    ...
}
 
- 首先这里把 
AbstractBeanFactory原来继承的DefaultSingletonBeanRegistry,修改为继承FactoryBeanRegistrySupport。- 因为需要扩展出创建 FactoryBean 对象的能力,所以这就想一个链条服务上,截出一个段来处理额外的服务,再把链条再链接上。
 
 - 此处新增加的功能主要是在 
doGetBean方法中,添加了调用(T) getObjectFromBeanInstance(sharedInstance, name)对获取FactoryBean的操作。 - 在 
getObjectFromBeanInstance方法中做具体的instanceof判断,另外还会从FactoryBean的缓存中获取对象,如果不存在则调用FactoryBeanRegistrySupport#getObjectFromFactoryBean,执行具体的操作。 
四、测试:对象作用域和FactoryBean
4.1 添加测试配置
4.1.1 添加IUserDao接口
IUserDao.java
package com.lino.springframework.test.bean;
/**
 * @description: 用户dao接口
 */
public interface IUserDao {
    /**
     * 根据用户ID获取用户名称
     *
     * @param uId 用户ID
     * @return 用户名称
     */
    String queryUserName(String uId);
}
 
- 定义一个 
IUserDao接口,是为了通过FactoryBean做一个自定义对象的代理操作。 
4.1.2 修改UserService用户对象
UserService.java
package com.lino.springframework.test.bean;
/**
 * @description: 模拟用户 Bean 对象
 */
public class UserService {
    private String uId;
    private String company;
    private String location;
    private IUserDao userDao;
    /**
     * 查询用户信息
     */
    public String queryUserInfo() {
        return userDao.queryUserName(uId) + "," + company + "," + location;
    }
    public String getuId() {
        return uId;
    }
    public void setuId(String uId) {
        this.uId = uId;
    }
    public String getCompany() {
        return company;
    }
    public void setCompany(String company) {
        this.company = company;
    }
    public String getLocation() {
        return location;
    }
    public void setLocation(String location) {
        this.location = location;
    }
    public IUserDao getUserDao() {
        return userDao;
    }
    public void setUserDao(IUserDao userDao) {
        this.userDao = userDao;
    }
}
 
- 在 UserService 添加属性为 
IUserDao,后面会给这个属性注入代理对象。 
4.1.3 定义FactoryBean对象
ProxyBeanFactory.java
package com.lino.springframework.test.bean;
import com.lino.springframework.beans.factory.FactoryBean;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy;
import java.util.HashMap;
import java.util.Map;
/**
 * @description: 代理对象工厂
 */
public class ProxyBeanFactory implements FactoryBean<IUserDao> {
    @Override
    public IUserDao getObject() throws Exception {
        InvocationHandler handler = (proxy, method, args) -> {
            // 添加排除方法
            if ("toString".equals(method.getName())) {
                return this.toString();
            }
            Map<String, String> hashMap = new HashMap<>();
            hashMap.put("10001", "张三");
            hashMap.put("10002", "李四");
            hashMap.put("10003", "王五");
            return "你被代理了 " + method.getName() + ": " + hashMap.get(args[0].toString());
        };
        return (IUserDao) Proxy.newProxyInstance(
                Thread.currentThread().getContextClassLoader(),
                new Class[]{IUserDao.class},
                handler);
    }
    @Override
    public Class<?> getObjectType() {
        return IUserDao.class;
    }
    @Override
    public boolean isSingleton() {
        return true;
    }
}
 
- 这是一个实现接口 
FactoryBean的代理类ProxyBeanFactory名称,主要是模拟了UserDao的原有功能,类似于 Mybatis 框架中的代理操作。 getObject()中提供的就是一个InvocationHandler的代理对象,当有方法调用的时候,则执行代理对象的功能。
4.1.4 配置文件
spring.xml
<?xml version="1.0" encoding="utf-8" ?>
<beans>
    <bean id="userService" class="com.lino.springframework.test.bean.UserService" scope="prototype">
        <property name="uId" value="10001"/>
        <property name="company" value="阿里"/>
        <property name="location" value="杭州"/>
        <property name="userDao" ref="proxyUserDao"/>
    </bean>
    <bean id="proxyUserDao" class="com.lino.springframework.test.bean.ProxyBeanFactory"/>
</beans>
 
- 在配置文件中,把 
proxyUserDao这个代理对象,注入到userService的userDao中。 
4.2 单元测试
4.2.1 单例&原型测试
ApiTest.java
@Test
public void test_prototype() {
    // 1.初始化 BeanFactory
    ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext("classpath:spring.xml");
    applicationContext.registerShutdownHook();
    // 2.获取Bean对象调用方法
    UserService userService01 = applicationContext.getBean("userService", UserService.class);
    UserService userService02 = applicationContext.getBean("userService", UserService.class);
    // 3.配置 scope="prototype/singleton"
    System.out.println(userService01);
    System.out.println(userService02);
    // 4.打印十六进制哈希
    System.out.println(userService01 + " 十六进制哈希: " + Integer.toHexString(userService01.hashCode()));
    System.out.println(ClassLayout.parseInstance(userService01).toPrintable());
}
 
- 在 
spring.xml配置文件中,设置了scope="prototype",这样就每次获取到的对象都应该是一个新的对象。 - 这里判断对象是否为一个单例,通过打印类对象的哈希值。
 
测试结果
com.lino.springframework.test.bean.UserService$$EnhancerByCGLIB$$5e50ce60@6bdf28bb
com.lino.springframework.test.bean.UserService$$EnhancerByCGLIB$$5e50ce60@6b71769e
com.lino.springframework.test.bean.UserService$$EnhancerByCGLIB$$5e50ce60@6bdf28bb 十六进制哈希: 6bdf28bb
com.lino.springframework.test.bean.UserService$$EnhancerByCGLIB$$5e50ce60 object internals:
 OFFSET  SIZE                                          TYPE DESCRIPTION                                               VALUE
      0     4                                               (object header)                                           01 bb 28 df (00000001 10111011 00101000 11011111) (-550978815)
      4     4                                               (object header)                                           6b 00 00 00 (01101011 00000000 00000000 00000000) (107)
      8     4                                               (object header)                                           3d e5 01 f8 (00111101 11100101 00000001 11111000) (-134093507)
     12     4                              java.lang.String UserService.uId                                           (object)
     16     4                              java.lang.String UserService.company                                       (object)
     20     4                              java.lang.String UserService.location                                      (object)
     24     4   com.lino.springframework.test.bean.IUserDao UserService.userDao                                       (object)
     28     1                                       boolean UserService$$EnhancerByCGLIB$$5e50ce60.CGLIB$BOUND        true
     29     3                                               (alignment/padding gap)
     32     4                       net.sf.cglib.proxy.NoOp UserService$$EnhancerByCGLIB$$5e50ce60.CGLIB$CALLBACK_0   (object)
     36     4                                               (loss due to the next object alignment)
Instance size: 40 bytes
Space losses: 3 bytes internal + 4 bytes external = 7 bytes total
 
4.2.2 代理对象测试
ApiTest.java
@Test
public void test_factory_bean() {
    // 1.初始化 BeanFactory
    ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext("classpath:spring.xml");
    applicationContext.registerShutdownHook();
    // 2.调用代理方法
    UserService userService = applicationContext.getBean("userService", UserService.class);
    System.out.println("测试结果:" + userService.queryUserInfo());
}
 
测试结果
测试结果:你被代理了 queryUserName: 张三,阿里,杭州
 
- 从测试结果看,代理类 
ProxyBeanFactory完美替换掉了UserDao的功能。 
五、总结:对象作用域和FactoryBean
- 在 Spring 框架整个开发的过程中,前期的各个功能接口类扩展的像膨胀了似的,但到后期在完善功能时,就没有那么难了,反而深入理解后会觉得功能的补充,都比较简单。只需要在所遇领域范围内进行扩展相应的服务实现即可。
 - 当你阅读完关于 
FactoryBean的实现以及测试过程的使用,以后再需要使用FactoryBean开发相应的组件时,一定会非常清楚它是如何创建自己的复杂 Bean 对象以及在什么时候初始化和调用的。遇到问题也可以快速的排查、定位和解决。 



















