横刀跃马,关于Bean对象作用域以及FactoryBean的实现和使用

开发 前端
交给 Spring 管理的 Bean 对象,一定就是我们用类创建出来的 Bean 吗?创建出来的 Bean 就永远是单例的吗,没有可能是原型模式吗?

[[408524]]

本文转载自微信公众号「bugstack虫洞栈」,作者小傅哥。转载本文请联系bugstack虫洞栈公众号。

 目录

  • 一、前言
  • 二、目标
  • 三、方案
  • 四、实现
    • 1. 工程结构
    • 2. Bean的作用范围定义和xml解析
    • 3. 创建和修改对象时候判断单例和原型模式
    • 4. 定义 FactoryBean 接口
    • 5. 实现一个 FactoryBean 注册服务
    • 6. 扩展 AbstractBeanFactory 创建对象逻辑
  • 五、测试
    • 1. 事先准备
    • 2. 定义 FactoryBean 对象
    • 3. 配置文件
    • 4. 单元测试(单例&原型)
    • 5. 单元测试(代理对象)
  • 六、总结

一、前言

老司机,你的砖怎么搬的那么快?

是有劲?是技巧?是后门?总之,那个老司机的代码总是可以很快的完成产品每次新增的需求,就像他俩是一家似的!而你就不一样了,不只产品经理还有运营、测试的小姐姐,都得给你买吃的,求着你赶紧把Bug修修,否则都来不及上线了。

那为啥别人的代码总是可以很快的扩展新功能,而你的代码从来不能被重构只能被重写,小需求小改、大需求大改,没需求呢?没需求自己跑着跑着也能崩溃,半夜被运维薅起来:“你这怎么又有数据库慢查询,把别人业务都拖拉胯了!”

有人说30岁的人都,还和刚毕业的做一样的活,是没进步的! 这太扯淡了,同样是同样的活,但做出来的结果可不一定是一样的,有人能用ifelse把产品功能凑出来,也有人可以把需求拆解成各个功能模块,定义接口、抽象类、实现和继承,运用设计模式构建出一套新增需求时候能快速实现,出现问题能准确定位的代码逻辑。这就像有人问:“树上有十只鸟,一枪打死一只,还有几只?”,你会想到什么?枪声大吗、鸟笼了吗、鸟被绑树上了吗、有鸟残疾的吗、鸟被打死了吗、打鸟的人眼睛好使吗、算肚子里怀孕的鸟吗、打鸟犯法吗、边上树还有其他鸟吗等等,这些都是一个职业技术人在一个行业磨练出来的经验,不是1天2天看几本书,刷几个洗脑文章能吸收的。

二、目标

交给 Spring 管理的 Bean 对象,一定就是我们用类创建出来的 Bean 吗?创建出来的 Bean 就永远是单例的吗,没有可能是原型模式吗?

在集合 Spring 框架下,我们使用的 MyBatis 框架中,它的核心作用是可以满足用户不需要实现 Dao 接口类,就可以通过 xml 或者注解配置的方式完成对数据库执行 CRUD 操作,那么在实现这样的 ORM 框架中,是怎么把一个数据库操作的 Bean 对象交给 Spring 管理的呢。

因为我们在使用 Spring、MyBatis 框架的时候都可以知道,并没有手动的去创建任何操作数据库的 Bean 对象,有的仅仅是一个接口定义,而这个接口定义竟然可以被注入到其他需要使用 Dao 的属性中去了,那么这一过程最核心待解决的问题,就是需要完成把复杂且以代理方式动态变化的对象,注册到 Spring 容器中。而为了满足这样的一个扩展组件开发的需求,就需要我们在现有手写的 Spring 框架中,添加这一能力。

三、方案

关于提供一个能让使用者定义复杂的 Bean 对象,功能点非常不错,意义也非常大,因为这样做了之后 Spring 的生态种子孵化箱就此提供了,谁家的框架都可以在此标准上完成自己服务的接入。

但这样的功能逻辑设计上并不复杂,因为整个 Spring 框架在开发的过程中就已经提供了各项扩展能力的接茬,你只需要在合适的位置提供一个接茬的处理接口调用和相应的功能逻辑实现即可,像这里的目标实现就是对外提供一个可以二次从 FactoryBean 的 getObject 方法中获取对象的功能即可,这样所有实现此接口的对象类,就可以扩充自己的对象功能了。MyBatis 就是实现了一个 MapperFactoryBean 类,在 getObject 方法中提供 SqlSession 对执行 CRUD 方法的操作 整体设计结构如下图:

  • 整个的实现过程包括了两部分,一个解决单例还是原型对象,另外一个处理 FactoryBean 类型对象创建过程中关于获取具体调用对象的 getObject 操作。
  • SCOPE_SINGLETON、SCOPE_PROTOTYPE,对象类型的创建获取方式,主要区分在于 AbstractAutowireCapableBeanFactory#createBean 创建完成对象后是否放入到内存中,如果不放入则每次获取都会重新创建。
  • createBean 执行对象创建、属性填充、依赖加载、前置后置处理、初始化等操作后,就要开始做执行判断整个对象是否是一个 FactoryBean 对象,如果是这样的对象,就需要再继续执行获取 FactoryBean 具体对象中的 getObject 对象了。整个 getBean 过程中都会新增一个单例类型的判断factory.isSingleton(),用于决定是否使用内存存放对象信息。

四、实现

1. 工程结构

  1. small-spring-step-09 
  2. └── src 
  3.     ├── main 
  4.     │   └── java 
  5.     │       └── cn.bugstack.springframework 
  6.     │           ├── beans 
  7.     │           │   ├── factory 
  8.     │           │   │   ├── config 
  9.     │           │   │   │   ├── AutowireCapableBeanFactory.java 
  10.     │           │   │   │   ├── BeanDefinition.java 
  11.     │           │   │   │   ├── BeanFactoryPostProcessor.java 
  12.     │           │   │   │   ├── BeanPostProcessor.java 
  13.     │           │   │   │   ├── BeanReference.java 
  14.     │           │   │   │   ├── ConfigurableBeanFactory.java 
  15.     │           │   │   │   └── SingletonBeanRegistry.java 
  16.     │           │   │   ├── support 
  17.     │           │   │   │   ├── AbstractAutowireCapableBeanFactory.java 
  18.     │           │   │   │   ├── AbstractBeanDefinitionReader.java 
  19.     │           │   │   │   ├── AbstractBeanFactory.java 
  20.     │           │   │   │   ├── BeanDefinitionReader.java 
  21.     │           │   │   │   ├── BeanDefinitionRegistry.java 
  22.     │           │   │   │   ├── CglibSubclassingInstantiationStrategy.java 
  23.     │           │   │   │   ├── DefaultListableBeanFactory.java 
  24.     │           │   │   │   ├── DefaultSingletonBeanRegistry.java 
  25.     │           │   │   │   ├── DisposableBeanAdapter.java 
  26.     │           │   │   │   ├── FactoryBeanRegistrySupport.java 
  27.     │           │   │   │   ├── InstantiationStrategy.java 
  28.     │           │   │   │   └── SimpleInstantiationStrategy.java   
  29.     │           │   │   ├── support 
  30.     │           │   │   │   └── XmlBeanDefinitionReader.java 
  31.     │           │   │   ├── Aware.java 
  32.     │           │   │   ├── BeanClassLoaderAware.java 
  33.     │           │   │   ├── BeanFactory.java 
  34.     │           │   │   ├── BeanFactoryAware.java 
  35.     │           │   │   ├── BeanNameAware.java 
  36.     │           │   │   ├── ConfigurableListableBeanFactory.java 
  37.     │           │   │   ├── DisposableBean.java 
  38.     │           │   │   ├── FactoryBean.java 
  39.     │           │   │   ├── HierarchicalBeanFactory.java 
  40.     │           │   │   ├── InitializingBean.java 
  41.     │           │   │   └── ListableBeanFactory.java 
  42.     │           │   ├── BeansException.java 
  43.     │           │   ├── PropertyValue.java 
  44.     │           │   └── PropertyValues.java  
  45.     │           ├── context 
  46.     │           │   ├── support 
  47.     │           │   │   ├── AbstractApplicationContext.java  
  48.     │           │   │   ├── AbstractRefreshableApplicationContext.java  
  49.     │           │   │   ├── AbstractXmlApplicationContext.java  
  50.     │           │   │   ├── ApplicationContextAwareProcessor.java  
  51.     │           │   │   └── ClassPathXmlApplicationContext.java  
  52.     │           │   ├── ApplicationContext.java  
  53.     │           │   ├── ApplicationContextAware.java  
  54.     │           │   └── ConfigurableApplicationContext.java 
  55.     │           ├── core.io 
  56.     │           │   ├── ClassPathResource.java  
  57.     │           │   ├── DefaultResourceLoader.java  
  58.     │           │   ├── FileSystemResource.java  
  59.     │           │   ├── Resource.java  
  60.     │           │   ├── ResourceLoader.java  
  61.     │           │   └── UrlResource.java 
  62.     │           └── utils 
  63.     │               └── ClassUtils.java 
  64.     └── test 
  65.         └── java 
  66.             └── cn.bugstack.springframework.test 
  67.                 ├── bean 
  68.                 │   ├── UserDao.java 
  69.                 │   └── UserService.java 
  70.                 └── ApiTest.java 

Spring 单例、原型以及 FactoryBean 功能实现类关系,如图 10-2

图 10-2

  • 以上整个类关系图展示的就是添加 Bean 的实例化是单例还是原型模式以及 FactoryBean 的实现。
  • 其实整个实现的过程并不复杂,只是在现有的 AbstractAutowireCapableBeanFactory 类以及继承的抽象类 AbstractBeanFactory 中进行扩展。
  • 不过这次我们把 AbstractBeanFactory 继承的 DefaultSingletonBeanRegistry 类,中间加了一层 FactoryBeanRegistrySupport,这个类在 Spring 框架中主要是处理关于 FactoryBean 注册的支撑操作。

2. Bean的作用范围定义和xml解析

  • cn.bugstack.springframework.beans.factory.config.BeanDefinition
  1. public class BeanDefinition { 
  2.  
  3.     String SCOPE_SINGLETON = ConfigurableBeanFactory.SCOPE_SINGLETON; 
  4.  
  5.     String SCOPE_PROTOTYPE = ConfigurableBeanFactory.SCOPE_PROTOTYPE; 
  6.  
  7.     private Class beanClass; 
  8.  
  9.     private PropertyValues propertyValues; 
  10.  
  11.     private String initMethodName; 
  12.  
  13.     private String destroyMethodName; 
  14.  
  15.     private String scope = SCOPE_SINGLETON; 
  16.  
  17.     private boolean singleton = true
  18.  
  19.     private boolean prototype = false
  20.      
  21.     // ...get/set 
  • singleton、prototype,是本次在 BeanDefinition 类中新增加的两个属性信息,用于把从 spring.xml 中解析到的 Bean 对象作用范围填充到属性中。

cn.bugstack.springframework.beans.factory.xml.XmlBeanDefinitionReader

  1. public class XmlBeanDefinitionReader extends AbstractBeanDefinitionReader { 
  2.  
  3.     protected void doLoadBeanDefinitions(InputStream inputStream) throws ClassNotFoundException { 
  4.        
  5.         for (int i = 0; i < childNodes.getLength(); i++) { 
  6.             // 判断元素 
  7.             if (!(childNodes.item(i) instanceof Element)) continue
  8.             // 判断对象 
  9.             if (!"bean".equals(childNodes.item(i).getNodeName())) continue
  10.  
  11.             // 解析标签 
  12.             Element bean = (Element) childNodes.item(i); 
  13.             String id = bean.getAttribute("id"); 
  14.             String name = bean.getAttribute("name"); 
  15.             String className = bean.getAttribute("class"); 
  16.             String initMethod = bean.getAttribute("init-method"); 
  17.             String destroyMethodName = bean.getAttribute("destroy-method"); 
  18.             String beanScope = bean.getAttribute("scope"); 
  19.  
  20.             // 获取 Class,方便获取类中的名称 
  21.             Class<?> clazz = Class.forName(className); 
  22.             // 优先级 id > name 
  23.             String beanName = StrUtil.isNotEmpty(id) ? id : name
  24.             if (StrUtil.isEmpty(beanName)) { 
  25.                 beanName = StrUtil.lowerFirst(clazz.getSimpleName()); 
  26.             } 
  27.  
  28.             // 定义Bean 
  29.             BeanDefinition beanDefinition = new BeanDefinition(clazz); 
  30.             beanDefinition.setInitMethodName(initMethod); 
  31.             beanDefinition.setDestroyMethodName(destroyMethodName); 
  32.  
  33.             if (StrUtil.isNotEmpty(beanScope)) { 
  34.                 beanDefinition.setScope(beanScope); 
  35.             } 
  36.              
  37.             // ... 
  38.              
  39.             // 注册 BeanDefinition 
  40.             getRegistry().registerBeanDefinition(beanName, beanDefinition); 
  41.         } 
  42.     } 
  43.  
  • 在解析 XML 处理类 XmlBeanDefinitionReader 中,新增加了关于 Bean 对象配置中 scope 的解析,并把这个属性信息填充到 Bean 定义中。beanDefinition.setScope(beanScope)

3. 创建和修改对象时候判断单例和原型模式

cn.bugstack.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory

  1. public abstract class AbstractAutowireCapableBeanFactory extends AbstractBeanFactory implements AutowireCapableBeanFactory { 
  2.  
  3.     private InstantiationStrategy instantiationStrategy = new CglibSubclassingInstantiationStrategy(); 
  4.  
  5.     @Override 
  6.     protected Object createBean(String beanName, BeanDefinition beanDefinition, Object[] args) throws BeansException { 
  7.         Object bean = null
  8.         try { 
  9.             bean = createBeanInstance(beanDefinition, beanName, args); 
  10.             // 给 Bean 填充属性 
  11.             applyPropertyValues(beanName, bean, beanDefinition); 
  12.             // 执行 Bean 的初始化方法和 BeanPostProcessor 的前置和后置处理方法 
  13.             bean = initializeBean(beanName, bean, beanDefinition); 
  14.         } catch (Exception e) { 
  15.             throw new BeansException("Instantiation of bean failed", e); 
  16.         } 
  17.  
  18.         // 注册实现了 DisposableBean 接口的 Bean 对象 
  19.         registerDisposableBeanIfNecessary(beanName, bean, beanDefinition); 
  20.  
  21.         // 判断 SCOPE_SINGLETON、SCOPE_PROTOTYPE 
  22.         if (beanDefinition.isSingleton()) { 
  23.             addSingleton(beanName, bean); 
  24.         } 
  25.         return bean; 
  26.     } 
  27.  
  28.     protected void registerDisposableBeanIfNecessary(String beanName, Object bean, BeanDefinition beanDefinition) { 
  29.         // 非 Singleton 类型的 Bean 不执行销毁方法 
  30.         if (!beanDefinition.isSingleton()) return
  31.  
  32.         if (bean instanceof DisposableBean || StrUtil.isNotEmpty(beanDefinition.getDestroyMethodName())) { 
  33.             registerDisposableBean(beanName, new DisposableBeanAdapter(bean, beanName, beanDefinition)); 
  34.         } 
  35.     } 
  36.      
  37.     // ... 其他功能 
  • 单例模式和原型模式的区别就在于是否存放到内存中,如果是原型模式那么就不会存放到内存中,每次获取都重新创建对象,另外非 Singleton 类型的 Bean 不需要执行销毁方法。
  • 所以这里的代码会有两处修改,一处是 createBean 中判断是否添加到 addSingleton(beanName, bean);,另外一处是 registerDisposableBeanIfNecessary 销毁注册中的判断 if (!beanDefinition.isSingleton()) return;。

4. 定义 FactoryBean 接口

cn.bugstack.springframework.beans.factory.FactoryBean

  1. public interface FactoryBean<T> { 
  2.  
  3.     T getObject() throws Exception; 
  4.  
  5.     Class<?> getObjectType(); 
  6.  
  7.     boolean isSingleton(); 
  8.  

FactoryBean 中需要提供3个方法,获取对象、对象类型,以及是否是单例对象,如果是单例对象依然会被放到内存中。

5. 实现一个 FactoryBean 注册服务

cn.bugstack.springframework.beans.factory.support.FactoryBeanRegistrySupport

  1. public abstract class FactoryBeanRegistrySupport extends DefaultSingletonBeanRegistry { 
  2.  
  3.     /** 
  4.      * Cache of singleton objects created by FactoryBeans: FactoryBean name --> object 
  5.      */ 
  6.     private final Map<String, Object> factoryBeanObjectCache = new ConcurrentHashMap<String, Object>(); 
  7.  
  8.     protected Object getCachedObjectForFactoryBean(String beanName) { 
  9.         Object object = this.factoryBeanObjectCache.get(beanName); 
  10.         return (object != NULL_OBJECT ? object : null); 
  11.     } 
  12.  
  13.     protected Object getObjectFromFactoryBean(FactoryBean factory, String beanName) { 
  14.         if (factory.isSingleton()) { 
  15.             Object object = this.factoryBeanObjectCache.get(beanName); 
  16.             if (object == null) { 
  17.                 object = doGetObjectFromFactoryBean(factory, beanName); 
  18.                 this.factoryBeanObjectCache.put(beanName, (object != null ? object : NULL_OBJECT)); 
  19.             } 
  20.             return (object != NULL_OBJECT ? object : null); 
  21.         } else { 
  22.             return doGetObjectFromFactoryBean(factory, beanName); 
  23.         } 
  24.     } 
  25.  
  26.     private Object doGetObjectFromFactoryBean(final FactoryBean factory, final String beanName){ 
  27.         try { 
  28.             return factory.getObject(); 
  29.         } catch (Exception e) { 
  30.             throw new BeansException("FactoryBean threw exception on object[" + beanName + "] creation", e); 
  31.         } 
  32.     } 
  33.  
  • FactoryBeanRegistrySupport 类主要处理的就是关于 FactoryBean 此类对象的注册操作,之所以放到这样一个单独的类里,就是希望做到不同领域模块下只负责各自需要完成的功能,避免因为扩展导致类膨胀到难以维护。
  • 同样这里也定义了缓存操作 factoryBeanObjectCache,用于存放单例类型的对象,避免重复创建。在日常使用用,基本也都是创建的单例对象
  • doGetObjectFromFactoryBean 是具体的获取 FactoryBean#getObject() 方法,因为既有缓存的处理也有对象的获取,所以额外提供了 getObjectFromFactoryBean 进行逻辑包装,这部分的操作方式是不和你日常做的业务逻辑开发非常相似。从Redis取数据,如果为空就从数据库获取并写入Redis

6. 扩展 AbstractBeanFactory 创建对象逻辑

cn.bugstack.springframework.beans.factory.support.AbstractBeanFactory

  1. public abstract class AbstractBeanFactory extends FactoryBeanRegistrySupport implements ConfigurableBeanFactory { 
  2.  
  3.     protected <T> T doGetBean(final String name, final Object[] args) { 
  4.         Object sharedInstance = getSingleton(name); 
  5.         if (sharedInstance != null) { 
  6.             // 如果是 FactoryBean,则需要调用 FactoryBean#getObject 
  7.             return (T) getObjectForBeanInstance(sharedInstance, name); 
  8.         } 
  9.  
  10.         BeanDefinition beanDefinition = getBeanDefinition(name); 
  11.         Object bean = createBean(name, beanDefinition, args); 
  12.         return (T) getObjectForBeanInstance(bean, name); 
  13.     }   
  14.     
  15.     private Object getObjectForBeanInstance(Object beanInstance, String beanName) { 
  16.         if (!(beanInstance instanceof FactoryBean)) { 
  17.             return beanInstance; 
  18.         } 
  19.  
  20.         Object object = getCachedObjectForFactoryBean(beanName); 
  21.  
  22.         if (object == null) { 
  23.             FactoryBean<?> factoryBean = (FactoryBean<?>) beanInstance; 
  24.             object = getObjectFromFactoryBean(factoryBean, beanName); 
  25.         } 
  26.  
  27.         return object; 
  28.     } 
  29.          
  30.     // ... 
  • 首先这里把 AbstractBeanFactory 原来继承的 DefaultSingletonBeanRegistry,修改为继承 FactoryBeanRegistrySupport。因为需要扩展出创建 FactoryBean 对象的能力,所以这就想一个链条服务上,截出一个段来处理额外的服务,并把链条再链接上。
  • 此处新增加的功能主要是在 doGetBean 方法中,添加了调用 (T) getObjectForBeanInstance(sharedInstance, name) 对获取 FactoryBean 的操作。
  • 在 getObjectForBeanInstance 方法中做具体的 instanceof 判断,另外还会从 FactoryBean 的缓存中获取对象,如果不存在则调用 FactoryBeanRegistrySupport#getObjectFromFactoryBean,执行具体的操作。

五、测试

1. 事先准备

cn.bugstack.springframework.test.bean.IUserDao

  1. public interface IUserDao { 
  2.  
  3.     String queryUserName(String uId); 
  4.  
  • 这个章节我们删掉 UserDao,定义一个 IUserDao 接口,之所这样做是为了通过 FactoryBean 做一个自定义对象的代理操作。

cn.bugstack.springframework.test.bean.UserService

  1. public class UserService { 
  2.  
  3.     private String uId; 
  4.     private String company; 
  5.     private String location; 
  6.     private IUserDao userDao; 
  7.  
  8.     public String queryUserInfo() { 
  9.         return userDao.queryUserName(uId) + "," + company + "," + location; 
  10.     } 
  11.  
  12.     // ...get/set 

在 UserService 新修改了一个原有 UserDao 属性为 IUserDao,后面我们会给这个属性注入代理对象。

2. 定义 FactoryBean 对象

cn.bugstack.springframework.test.bean.ProxyBeanFactory

  1. public class ProxyBeanFactory implements FactoryBean<IUserDao> { 
  2.  
  3.     @Override 
  4.     public IUserDao getObject() throws Exception { 
  5.         InvocationHandler handler = (proxy, method, args) -> { 
  6.  
  7.             Map<String, String> hashMap = new HashMap<>(); 
  8.             hashMap.put("10001""小傅哥"); 
  9.             hashMap.put("10002""八杯水"); 
  10.             hashMap.put("10003""阿毛"); 
  11.              
  12.             return "你被代理了 " + method.getName() + ":" + hashMap.get(args[0].toString()); 
  13.         }; 
  14.         return (IUserDao) Proxy.newProxyInstance(Thread.currentThread().getContextClassLoader(), new Class[]{IUserDao.class}, handler); 
  15.     } 
  16.  
  17.     @Override 
  18.     public Class<?> getObjectType() { 
  19.         return IUserDao.class; 
  20.     } 
  21.  
  22.     @Override 
  23.     public boolean isSingleton() { 
  24.         return true
  25.     } 
  26.  
  • 这是一个实现接口 FactoryBean 的代理类 ProxyBeanFactory 名称,主要是模拟了 UserDao 的原有功能,类似于 MyBatis 框架中的代理操作。
  • getObject() 中提供的就是一个 InvocationHandler 的代理对象,当有方法调用的时候,则执行代理对象的功能。

3. 配置文件

  1. <?xml version="1.0" encoding="UTF-8"?> 
  2. <beans> 
  3.  
  4.     <bean id="userService" class="cn.bugstack.springframework.test.bean.UserService" scope="prototype"
  5.         <property name="uId" value="10001"/> 
  6.         <property name="company" value="腾讯"/> 
  7.         <property name="location" value="深圳"/> 
  8.         <property name="userDao" ref="proxyUserDao"/> 
  9.     </bean> 
  10.  
  11.     <bean id="proxyUserDao" class="cn.bugstack.springframework.test.bean.ProxyBeanFactory"/> 
  12.  
  13. </beans> 
  • 在配置文件中,我们把 proxyUserDao 这个代理对象,注入到 userService 的 userDao 中。与上一章节相比,去掉了 UserDao 实现类,转而用代理类替换

4. 单元测试(单例&原型)

  1. @Test 
  2. public void test_prototype() { 
  3.     // 1.初始化 BeanFactory 
  4.     ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext("classpath:spring.xml"); 
  5.     applicationContext.registerShutdownHook();    
  6.  
  7.     // 2. 获取Bean对象调用方法 
  8.     UserService userService01 = applicationContext.getBean("userService", UserService.class); 
  9.     UserService userService02 = applicationContext.getBean("userService", UserService.class); 
  10.      
  11.     // 3. 配置 scope="prototype/singleton" 
  12.     System.out.println(userService01); 
  13.     System.out.println(userService02);     
  14.  
  15.     // 4. 打印十六进制哈希 
  16.     System.out.println(userService01 + " 十六进制哈希:" + Integer.toHexString(userService01.hashCode())); 
  17.     System.out.println(ClassLayout.parseInstance(userService01).toPrintable()); 
  18.  
  • 在 spring.xml 配置文件中,设置了 scope="prototype" 这样就每次获取到的对象都应该是一个新的对象。
  • 这里判断对象是否为一个会看到打印的类对象的哈希值,所以我们把十六进制哈希打印出来。

测试结果

  1. cn.bugstack.springframework.test.bean.UserService$$EnhancerByCGLIB$$4cabb984@1b0375b3 
  2. cn.bugstack.springframework.test.bean.UserService$$EnhancerByCGLIB$$4cabb984@2f7c7260 
  3. cn.bugstack.springframework.test.bean.UserService$$EnhancerByCGLIB$$4cabb984@1b0375b3 十六进制哈希:1b0375b3 
  4. cn.bugstack.springframework.test.bean.UserService$$EnhancerByCGLIB$$4cabb984 object internals: 
  5.  OFFSET  SIZE                                             TYPE DESCRIPTION                                               VALUE 
  6.       0     4                                                  (object header)                                           01 b3 75 03 (00000001 10110011 01110101 00000011) (58045185) 
  7.       4     4                                                  (object header)                                           1b 00 00 00 (00011011 00000000 00000000 00000000) (27) 
  8.       8     4                                                  (object header)                                           9f e1 01 f8 (10011111 11100001 00000001 11111000) (-134094433) 
  9.      12     4                                 java.lang.String UserService.uId                                           (object) 
  10.      16     4                                 java.lang.String UserService.company                                       (object) 
  11.      20     4                                 java.lang.String UserService.location                                      (object) 
  12.      24     4   cn.bugstack.springframework.test.bean.IUserDao UserService.userDao                                       (object) 
  13.      28     1                                          boolean UserService$$EnhancerByCGLIB$$4cabb984.CGLIB$BOUND        true 
  14.      29     3                                                  (alignment/padding gap)                                   
  15.      32     4                          net.sf.cglib.proxy.NoOp UserService$$EnhancerByCGLIB$$4cabb984.CGLIB$CALLBACK_0   (object) 
  16.      36     4                                                  (loss due to the next object alignment) 
  17. Instance size: 40 bytes 
  18. Space losses: 3 bytes internal + 4 bytes external = 7 bytes total 
  19.  
  20.  
  21. Process finished with exit code 0 

  • 对象后面的这一小段字符串就是16进制哈希值,在对象头哈希值存放的结果上看,也有对应的数值。只不过这个结果是倒过来的。
  • 另外可以看到 cabb984@1b0375b3、cabb984@2f7c7260,这两个对象的结尾16进制哈希值并不一样,所以我们的原型模式是生效的。

5. 单元测试(代理对象)

  1. @Test 
  2. public void test_factory_bean() { 
  3.     // 1.初始化 BeanFactory 
  4.     ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext("classpath:spring.xml"); 
  5.     applicationContext.registerShutdownHook();  
  6.  
  7.     // 2. 调用代理方法 
  8.     UserService userService = applicationContext.getBean("userService", UserService.class); 
  9.     System.out.println("测试结果:" + userService.queryUserInfo()); 

关于 FactoryBean 的调用并没有太多不一样,因为所有的不同都已经被 spring.xml 配置进去了。当然你可以直接调用 spring.xml 配置的对象 cn.bugstack.springframework.test.bean.ProxyBeanFactory

测试结果

测试结果:你被代理了 queryUserName:小傅哥,腾讯,深圳

Process finished with exit code 0

  • 从测试结果来看,我们的代理类 ProxyBeanFactory 已经完美替换掉了 UserDao 的功能。
  • 虽然看上去这一点实现并不复杂,甚至有点简单。但就是这样一点点核心内容的设计了,解决了所有需要和 Spring 结合的其他框架交互链接问题。如果对此类内容感兴趣,也可以阅读小傅哥《中间件设计和开发》

六、总结

  • 在 Spring 框架整个开发的过程中,前期的各个功能接口类扩展的像膨胀了似的,但到后期在完善功能时,就没有那么难了,反而深入理解后会觉得功能的补充,都比较简单。只需要再所遇领域范围内进行扩展相应的服务实现即可。
  • 当你仔细阅读完关于 FactoryBean 的实现以及测试过程的使用,以后再需要使用 FactoryBean 开发相应的组件时候,一定会非常清楚它是如何创建自己的复杂 Bean 对象以及在什么时候初始化和调用的。遇到问题也可以快速的排查、定位和解决。
  • 如果你在学习的过程中感觉这些类、接口、实现、继承,穿梭的很复杂,一时半会脑子还反应不过来。那么你最好的方式是动手去画画这些类关系图,梳理下实现的结构,看看每个类在干什么。

 

责任编辑:武晓燕 来源: bugstack虫洞栈
相关推荐

2011-03-18 09:27:00

Spring

2021-07-05 08:43:46

Spring Beanscope作用域

2023-06-29 08:32:41

Bean作用域

2023-09-05 08:23:56

SpringScope方法

2010-09-27 13:21:02

DHCP作用域

2010-09-01 09:10:30

DHCP作用域

2021-03-09 08:35:51

JSS作用域前端

2019-03-13 08:00:00

JavaScript作用域前端

2011-09-06 09:56:24

JavaScript

2021-06-03 07:55:12

技术

2024-01-05 08:38:20

SpringBeanScope

2021-07-01 11:56:51

JavaScript开发代码

2022-08-31 07:04:50

Bean作用域

2017-09-14 13:55:57

JavaScript

2022-08-17 08:10:34

语言VisitorListener

2009-01-04 09:08:30

面向对象继承接口

2009-06-12 09:49:25

EJB事务属性EJB事物

2021-04-14 07:52:00

Vue 作用域插槽

2021-12-06 07:15:48

Javascript作用域闭包

2023-07-06 14:24:23

Spring接口自定义
点赞
收藏

51CTO技术栈公众号