利用Spring IoC 容器进行项目开发的一个重要工作就是在 applicationContext.xml 中进行bean管理。然而,业务bean(BO或Service),DAO bean的配置量随项目的功能规模增长而增长,对于大型项目而言,庞大的 bean配置文件维护起来极为痛苦。例如:
-
- <bean id="authService" parent="baseTransactionProxy">
- <property name="target">
- <bean class="com.sr.jsfuser.service.AuthorityService">
- <property name="authDao" ref="authDao" />
- </bean>
- </property>
- </bean>
-
- <bean id="userService" parent="baseTransactionProxy">
- <property name="target">
- <bean class="com.sr.jsfuser.service.pojo.UserService">
- <property name="userDao" ref="userDao" />
- </bean>
- </property>
- </bean>
-
- <bean id="authDao" class="com.sr.jsfuser.dao.jdbc.AuthorityDao">
- <property name="dataSource" ref="dataSource" />
- </bean>
-
- <bean id="userDao" class="com.sr.jsfuser.dao.jdbc.UserDao">
- <property name="dataSource" ref="dataSource" />
- </bean>
上例为两个功能的配置,鉴权(auth)和用户(user)。这种Service bean和DAO bean 的管理方式是按功能的管理方式,即one by one,配置量是
F=f(n)
其中 n 是项目中功能的数量。上例虽充分体现了Spring IoC容器依赖注入的优良特性,但配置量过大,维护和管理难度很大,不是一种最佳实践。
2. 零配置的动态Service Bean管理(ServiceFactory)
有没有一种既能发挥Spring优势,又能节省配置的模式呢?我们可以观察到,每个功能的Service bean的配置,只有 dao 注入是不同的,其他如Service类的命名,Service的事务管理器注入,Service类和DAO类的命名关系等都具有相似性或者完全相同;dao bean配置类似。
因此,可以采用动态bean管理(Dynamic Bean Management)思想简化业务bean配置。动态bean管理使用新定义的 ServiceFactory 的createService方法来创建业务类Service实例, ServiceFactory 在创建时,自动进行dao创建,datasource注入等,不再需要进行bean的配置。原理如下图所示。
ServiceFactory原理有以下要点:
1) 利用了Spring ApplicationContext 的 registerBeanDefinition方法和动态bean注册技术;
2) 作为bean管理的一个规范:要求Service类实现 setDAO(DAO dao)方法,以便统一注入dao实例;
3) 作为bean管理的一个规范:调用 dao.setDatasource(datasource)方法进行数据源注入;
4) 作为bean管理的一个规范:内部使用className作为 service bean 和dao bean的内部id;
5) 作为bean管理的一个规范:ServiceFactory内获取bean实例时,都采用prototype scope。
ServiceFactory内部实现序列图如下:
使用ServiceFactory 进行业务开发,代码如下:
至于事务管理,即可使用向service注入transactionManager; 也可以使用 Spring 的
org.springframework.transaction.interceptor.TransactionInterceptor ,
org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator,
org.springframework.aop.support.NameMatchMethodPointcutAdvisor
三者配合,将 *Service类的 特定方法纳入事务范围。我们项目中使用的是后者,更为简便和松耦.
3. 通用的动态Bean管理(自定义BeanFactory类)
在上例中,注意新增了一个BeanFactory,区别于Spring 的 BeanFactory 接口。自定义 BeanFactory 通过 createBean(className) 进行动态bean注册和创建,
Object bean = BeanFactory.createBean(UserService.class);
相当于替代了Object bean = new UserService();
这相当于替代了 Java语言级的 new Object() ,并且使得类实例和Spring容器关联了起来,这样使用Spring的AOP配置就可以极为方便的对bean进行控制。(而上例中的ServiceFactory是该项目中利用BeanFactory的特定封装,其他项目的ServiceFactory实现可能不同,可参考实现)
4. JDK问题
以上方案主要针对1.4。 但同样适用1.5及以上。有朋友说,1.5中用 annotation + <context:component-scan/>标签也能完成类似的工作,但那样需要在 Java 代码中写很多 annotation , 个人不偏好使用 annotation ,因为它相当于xml配置移到了 Java代码中,并且维护不方便。
相关推荐
spring bean配置 运行环境:eclipse 构建工具:maven 不提供maven构建,maven用来解决jar包的依赖
普元eos-springbean开发
Spring数据库开发--SpringJDBC的配置
07 源码分析-spring的bean工厂准备工作
07 源码分析-spring的bean工厂准备工作
这个类库也提供一个简单的方式来注入 MyBatis 数据映射器和 SqlSession 到业务层的 bean 中。 而且它也会处理事务, 翻译 MyBatis 的异常到 Spring 的 DataAccessException 异常(数据访问异常,译者注)中。最终,它并 ...
Spring的Bean配置Spring的Bean配置说明说明
Spring--2.Spring 中的 Bean 配置-4
Spring--2.Spring 中的 Bean 配置-3
Spring--2.Spring 中的 Bean 配置-1
Spring--2.Spring 中的 Bean 配置-2-2
Spring--2.Spring 中的 Bean 配置-2-1
开发工具 spring-beans-4.3.6.RELEASE开发工具 spring-beans-4.3.6.RELEASE开发工具 spring-beans-4.3.6.RELEASE开发工具 spring-beans-4.3.6.RELEASE开发工具 spring-beans-4.3.6.RELEASE开发工具 spring-beans-...
04-spring5-spring-bean
Spring4--2.bean注解和AOP
这个类库也提供一个简单的方式来注入 MyBatis 数据映射器和 SqlSession 到业务层的 bean 中。 而且它也会处理事务, 翻译 MyBatis 的异常到 Spring 的 DataAccessException 异常(数据访问异常,译者注)中。最终,它并 ...
Java-Spring-WebService最基础的配置示例.txt
Spring IOC和DI概述,Bean的配置形式,IOC容器BeanFactory和ApplicationContext概述,依赖注入的方式,属性注入,构造器注入等案例
spring注解开发使用教程--Bean加载控制(二)案例代码
Java-Spring-SpringIoC容器-SpringIoC的学习 SpringIoC容器的学习笔记 主要介绍了IoC容器工作原理以及如何配置IoC容器 通过<bean>标签和注解配置容器