如何把Bean塞到Spring容器
1. 定义接口
1 2 3 4 5
| public interface IUserDao {
String queryUserInfo();
}
|
- 先定义一个类似 DAO 的接口,基本这样的接口在使用 MyBatis 时还是非常常见的。后面我们会对这个接口做代理和注册。
2. 类代理实现
1 2 3 4 5 6 7 8
| ClassLoader classLoader = Thread.currentThread().getContextClassLoader(); Class<?>[] classes = {IUserDao.class};
InvocationHandler handler = (proxy, method, args) -> "你被代理了 " + method.getName(); IUserDao userDao = (IUserDao) Proxy.newProxyInstance(classLoader, classes, handler);
String res = userDao.queryUserInfo(); logger.info("测试结果:{}", res);
|
- Java 本身的代理方式使用起来还是比较简单的,用法也很固定。
- InvocationHandler 是个接口类,它对应的实现内容就是代理对象的具体实现。
- 最后就是把代理交给 Proxy 创建代理对象,
Proxy.newProxyInstance
。
3. 实现Bean工厂
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
| public class ProxyBeanFactory implements FactoryBean {
@Override public Object getObject() throws Exception {
ClassLoader classLoader = Thread.currentThread().getContextClassLoader(); Class[] classes = {IUserDao.class}; InvocationHandler handler = (proxy, method, args) -> "你被代理了 " + method.getName();
return Proxy.newProxyInstance(classLoader, classes, handler); }
@Override public Class<?> getObjectType() { return IUserDao.class; }
}
|
- FactoryBean 在 spring 起到着二当家的地位,它将近有70多个小弟(实现它的接口定义),那么它有三个方法;
- T getObject() throws Exception; 返回bean实例对象
- Class<?> getObjectType(); 返回实例类类型
- boolean isSingleton(); 判断是否单例,单例会放到Spring容器中单实例缓存池中
- 在这里我们把上面使用Java代理的对象放到了 getObject() 方法中,那么现在再从 Spring 中获取到的对象,就是我们的代理对象了。
4. Bean 注册
1 2 3 4 5 6 7 8 9 10 11 12 13
| public class RegisterBeanFactory implements BeanDefinitionRegistryPostProcessor {
@Override public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException {
GenericBeanDefinition beanDefinition = new GenericBeanDefinition(); beanDefinition.setBeanClass(ProxyBeanFactory.class);
BeanDefinitionHolder definitionHolder = new BeanDefinitionHolder(beanDefinition, "userDao"); BeanDefinitionReaderUtils.registerBeanDefinition(definitionHolder, registry); }
}
|
在 Spring 的 Bean 管理中,所有的 Bean 最终都会被注册到类 DefaultListableBeanFactory 中,以上这部分代码主要的内容包括:
- 实现 BeanDefinitionRegistryPostProcessor.postProcessBeanDefinitionRegistry方法,获取 Bean 注册对象。
- 定义 Bean,GenericBeanDefinition,这里主要设置了我们的代理类工厂。
- 创建 Bean 定义处理类,BeanDefinitionHolder,这里需要的主要参数;定义 Bean 和名称
setBeanClass(ProxyBeanFactory.class)
。
- 最后将我们自己的bean注册到spring容器中去,registry.registerBeanDefinition()
四、测试验证
在上面我们已经把自定义代理的 Bean 注册到了 Spring 容器中,接下来我们来测试下这个代理的 Bean 被如何调用。
1. 定义 spring-config.xml
1
| <bean id="userDao" class="org.itstack.interview.bean.RegisterBeanFactory"/>
|
- 这里我们把 RegisterBeanFactory 配置到 spring 的 xml 配置中,便于启动时加载。
2. 单元测试
1 2 3 4 5 6 7
| @Test public void test_IUserDao() { BeanFactory beanFactory = new ClassPathXmlApplicationContext("spring-config.xml"); IUserDao userDao = beanFactory.getBean("userDao", IUserDao.class); String res = userDao.queryUserInfo(); logger.info("测试结果:{}", res); }
|
测试结果
1 2 3 4
| 22:53:14.760 [main] DEBUG o.s.b.f.s.DefaultListableBeanFactory - Returning cached instance of singleton bean 'userDao' 22:53:14.796 [main] INFO org.itstack.interview.test.ApiTest - 测试结果:你被代理了 queryUserInfo
Process finished with exit code 0
|
- 从测试结果可以看到,我们已经可以通过注入到Spring的代理Bean对象,实现我们的预期结果。
- 其实这个过程也是很多框架中用到的方式,尤其是在一些中间件开发,类似的 ORM 框架都需要使用到。