以多模块的形式组织项目,乃是maven一直提倡的方式。如果使用得当,多模块可以可以帮助项目划分模块,鼓励重用,防止POM变得过于庞大,方便某个模块的构建,而不用每次都构建整个项目,并且使得针对某个模块的特殊控制更为方便(关于多模块优势的描述可以参见juvenshun的博文Maven最佳实践:划分模块http://juvenshun.iteye.com/blog/305865。)
虽然多模块能给项目带来诸多好处,单实际操作时常常遇到一些各种各样的增加项目复杂度的问题,导致开发的效率下降。这里总结了一些在多模块项目实践中的常见的问题和解决办法,帮助降低复杂度,减少程序员的无谓劳动。
- 组织依赖
在多模块项目中,不同的模块依赖的包各自不同,而这些依赖中又往往会有一些相同的依赖。一种常见的组织方式是,每个程序员在自己负责的子模块的pom文件中加入该模块需要的依赖,然后往下进行开发。这样做会出现项目对同一个类库依赖各种不同的版本的情况,给项目带来了复杂度,同时也容易出现各种各样的问题。
最佳实践应该是,将可能被多个项目共同依赖的的包以及maven插件设置放在父模块的pom文件中,以dependencyManagement和pluginManagement的形式进行管理。这样做,一来实现了依赖的统一管理,二来统一了项目对类库依赖的版本,减少对程序员该来的迷惑和思考。
- 子模块间依赖
对于项目子模块间依赖,可以选择放到dependencyManagement中,也可以放在各自子模块的pom文件中,但需要注意的一点是,每个子模块的版本最好和整个项目的版本统一,通过在父模块pom的properties配置中设置project.version属性,各子模块依赖版本均以此为准的方式来处理。否则子模块的升级会带来各种意想不到的情况。
- 版本控制
Web项目中常用到spring框架,该框架的依赖可分为spring-core,spring-beans等等若干个,它们的版本号往往是统一的。如果我们因为某些原因需要改变spring的版本,需要将所有依赖的版本都进行修改,这样显然不是明智的做法。减少重复劳动的方法是,将重复存在的版本抽取出来,统一放在父模块pom的properties配置中统一管理。当日后需要变更版本时,只需要对父模块pom进行修改,便可到处生效。
另一个更进一步的做法是,新建一个子模块spring-dep,设置打包方式为pom,将spring相关的依赖全部放到这个子模块中,当项目需要使用spring时,只需要添加对该模块的依赖即可。这样做的好处是,spring-dep模块不仅可以在当前项目中使用,还可以放到其他的项目中,一劳永逸的解决了spring的版本问题。以此类推,也可以用这种方式对测试等等较为通用的框架依赖进行管理。
- 排除单个包依赖
在使用日志系统时,如果使用到了slf4j框架,则在依赖中需要将comons-logging依赖排除,避免其和JCL-over-SLF4J发生冲突,导致不能正常打日志。为了解决这个问题,在maven仓库中对commons-logging存在99.0-does-not-exist这个版本,该版本其实空无一物,因此依赖该版本相当于什么都没有依赖。当需要排除这个依赖时,则在父模块的pom中依赖commons-logging,并将版本设为99.0-does-not-exist,即可排除该依赖。
这是一个较为特别的排除依赖的方法,那么,除此之外还有没有更通用一些的手段呢?答案是有的,在父模块的pom文件的dependency中设置对要排除的包的依赖,然后将其scope设置为provided,根据依赖传递性,所有子模块中对该包的依赖都以父模块pom中的定义为准,所以最终依赖将不会包含该包,也就达到了排除依赖的目的。
这样做虽然能够排除依赖,不过需要注意的一点是,在单元测试过程中,该依赖仍然存在,会对单元测试造成影响。
- 多个spring子模块
在开发web项目时,通常会将项目分为project-web,project-service,project-dao等等若干个子模块。这时,可能需要在多个子模块中使用到spring框架的IoC机制,定义一些特定功能的bean。那我们如何在web子模块中识别并获取到其他子模块(比如service)的bean呢?首先,我们从部署的角度来看,无论是单模块项目,或者是多模块项目,在项目编译完成后,通常会被整体打包到一个war包中,然后将其部署到Web容器中运行,容器通过唯一的web.xml文件作为web工程配置的入口,执行这个web项目。因此,我们在需要使用spring框架的模块中配置好相应的applicationContext.xml文件(要放到src/main/resources下,这样能通过classpath轻松的找到),然后在web子模块的web.xml文件中引用这些所有的applicationContext文件。这样一来,所有的bean都能在web子模块中被识别和调用。
如果使用spring注解来注入类,那么一个比较好的实践是,对所有子模块的包命名相同的前缀(比如com.company.product.*),然后在web子模块的applicationContext文件中使用<context:component-scan base-package="com.company.product.*"/>,这里就能够扫描到所有com.company.product.*包下的component/controller/service,不用再在各个子模块中定义applicationContext文件。
- 使用maven filter过滤WEB-INF目录下文件
Maven filter能够过滤资源(resource)文件,通过在pom文件中配置:
<filters>
<filter>src/main/resources/config.properties</filter>
</filters>
<resources>
<resource>
<directory>src/main/resources</directory>
<filtering>true</filtering>
</resource>
</resources>
根据
properties
文件替换资源文件中的占位符。
而当需要过滤src/main/webapp/WEB-INF/目录下的文件(比如web.xml)时,不能在这里配置,因为这里是处理资源文件的,最终这些资源文件都会被放到target/classes目录下,不能满足需求,因此需要再另外做配置:
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-war-plugin</artifactId>
<configuration>
<webResources>
<webResource>
<!-- 过滤的目录-->
<directory>${basedir}/src/main/webapp/WEB-INF</directory>
<includes>
<!-- 添加需要过滤的文件 ,支持通配符-->
<include>web.xml</include>
</includes>
<!-- 打包之后目标目录 -->
<targetPath>WEB-INF</targetPath>
<filtering>true</filtering>
</webResource>
</webResources>
</configuration>
</plugin>