spring 分模块开发
背景 我们一般采用spring+struts+ibatis的开发模式,使用spring统管业务bean,当项目很大时,面临一个bean命名冲突问题,如果和struts一样,采取分模块的方式,不同的模块有自己的命名空间,将会避免几个并行开发的子系统或模块共用一套bean配置文件时id产生重名的问题。
简单方案 整个系统所使用的公共bean,例如数据源、事务管理、iBatis或hibernate的引擎、国际化资源文件等,放在一个root context文件中, 构造一个根ApplicationContext实例,作为各子模块的父,这里就看出Spring设计上的易扩展性了,ApplicationContext的构造子允许通过一个父实例来创建。 核心类 ContextHelper ,作为管理器类,如下: public class ContextHelper { private static Map _contexts ; private static ApplicationContext _rootContext; private static final String ROOT_CONTEXT = "conf/rootContext.xml"; static { _contexts = new HashMap(); try{ _rootContext = new ClassPathXmlApplicationContext(ROOT_CONTEXT); } catch (Exception e) { throw new ContextException(e); } }
public static synchronized Object getBean(String configName, String beanName) { ApplicationContext context = getApplicationContext(configName); Object bean = context.getBean(beanName); return bean; } //lazyload private static synchronized ApplicationContext getApplicationContext(String configName) { if(_contexts.containsKey(configName)) { return (ApplicationContext) _contexts.get(configName); } //这里规定了模块配置文件的命名规则 String moduleConfigFile = "conf/" + configName + ".bean.xml"; String [] configs = { moduleConfigFile}; ApplicationContext context = new ClassPathXmlApplicationContext(configs,_rootContext); _contexts.put(configName, context); return context; } //消息资源文件管理
public static String getMessage(String key,Object[] arg,Locale l){ return _rootContext.getMessage(key,arg,l); } }
需要注意的一点是,各子模块bean配置文件中,如果引用根配置中的bean,不可以使用local属性了,请使用bean=“dataSource“,一开始没注意到这个问题,总报错,无法正确合并,我调试了好久才发现。
ToDo
上述方案还比较简陋,主要的隐患在于,各模块之间的通信问题,当前的设计比较适用于各模块耦合度较小的情况,如果有一些bean,供几个模块之间共用,势必也需要提至rootContext.xml中配置,大大增加了根文件的复杂性。 另外,可供改进处还有一个,每个模块应该允许采用多个bean配置文件,允许将dao、service等tier分开来处理,可以通过建立一个properties文件,将模块逻辑名与若干配置文件建立映射关系来解决。我得承认,当前的简版方案在context load上有一些hard code的坏气味。 
|