maven学习(下)利用Profile构建不同环境的部署包

http://my.oschina.net/u/1866821/blog/464183



上回继续,项目开发好以后,通常要在多个环境部署,象我们公司多达5种环境:本机环境(local)、(开发小组内自测的)开发环境(dev)、(提供给测试团队的)测试环境(test)、预发布环境(pre)、正式生产环境(prod),每种环境都有各自的配置参数,比如:数据库连接、远程调用的ws地址等等。如果每个环境build前手动修改这些参数,显然太不fashion.

maven早就考虑到了这些问题,看下面的pom片段:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
< profiles >
         < profile >
             <!-- 本地环境 -->
             < id >local</ id >
             < properties >                
                 < db-url >jdbc:oracle:thin:@localhost:1521:XE</ db-url >
                 < db-username >***</ db-username >
                 < db-password >***</ db-password >
             </ properties >
         </ profile >
         < profile >
             <!-- 开发环境 -->
             < id >dev</ id >
             < properties >                
                 < db-url >jdbc:oracle:thin:@172.21.129.51:1521:orcl</ db-url >
                 < db-username >***</ db-username >
                 < db-password >***</ db-password >
             </ properties >
             <!-- 默认激活本环境 -->
             < activation >
                 < activeByDefault >true</ activeByDefault >
             </ activation >
         </ profile >
         ...
     </ profiles >

profiles节点中,定义了二种环境:local、dev(默认激活dev环境),可以在各自的环境中添加需要的property值,接下来修改build节点,参考下面的示例:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
< build >
         < resources >
             < resource >
                 < directory >src/main/resources</ directory >
                 < filtering >true</ filtering >
             </ resource >
         </ resources >
         < plugins >
             < plugin >
                 < groupId >org.apache.maven.plugins</ groupId >
                 < artifactId >maven-compiler-plugin</ artifactId >
                 < version >2.5.1</ version >
                 < configuration >
                     < source >1.6</ source >
                     < target >1.6</ target >
                     < encoding >utf-8</ encoding >
                 </ configuration >
             </ plugin >
         </ plugins >
     </ build >

resource节点是关键,它表明了哪个目录下的配置文件(不管是xml配置文件,还是properties属性文件),需要根据profile环境来替换属性值。

通常配置文件放在resources目录下,build时该目录下的文件都自动会copy到class目录下

以上图为例,其中spring-database.xml的内容为:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
<? xml  version = "1.0"  encoding = "UTF-8" ?>
< beans  xmlns = "http://www.springframework.org/schema/beans"
     xmlns:xsi = "http://www.w3.org/2001/XMLSchema-instance"
     xsi:schemaLocation="http://www.springframework.org/schema/beans 
     http://www.springframework.org/schema/beans/spring-beans.xsd">
 
     < bean  id = "dataSource"
         class = "org.springframework.jdbc.datasource.DriverManagerDataSource" >
         < property  name = "driverClassName"  value = "oracle.jdbc.driver.OracleDriver"  />
         < property  name = "url"  value = "${db-url}"  />
         < property  name = "username"  value = "${db-username}"  />
         < property  name = "password"  value = "${db-password}"  />        
     </ bean >
</ beans >

各属性节点的值,用占位符"${属性名}"占位,maven在package时,会根据profile的环境自动替换这些占位符为实际属性值。

默认情况下: 

maven package

将采用默认激活的profile环境来打包,也可以手动指定环境,比如:

maven package -P dev

将自动打包成dev环境的部署包(注:参数P为大写)

 

最后再给2个实例的运用例子:

1、开发环境与生产环境数据源采用不同方式的问题

本机开发时为了方便,很多开发人员喜欢直接用JDBC直接连接数据库,这样修改起来方便;

?
1
2
3
4
5
6
7
8
9
10
11
< bean  id = "dataSource"  class = "org.apache.commons.dbcp.BasicDataSource"
         destroy-method = "close" >
         < property  name = "driverClassName"  value = "oracle.jdbc.driver.OracleDriver"  />
         < property  name = "url"  value = "${db-url}"  />
         < property  name = "username"  value = "${db-username}"  />
         < property  name = "password"  value = "${db-password}"  />
         < property  name = "defaultAutoCommit"  value = "false"  />
         < property  name = "initialSize"  value = "2"  />
         < property  name = "maxActive"  value = "10"  />
         < property  name = "maxWait"  value = "60000"  />
</ bean >

而生产环境,通常是在webserver(比如weblogic上)配置一个JNDI数据源,

?
1
2
3
< bean  id = "dataSource"  class = "org.springframework.jndi.JndiObjectFactoryBean" >
     < property  name = "jndiName"  value = "appDS"  />
</ bean >

如果每次发布生产前,都要手动修改,未免太原始,可以通过maven的profile来解决

先把配置文件改成 

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
< bean  id = "${db-source-jdbc}"  class = "org.apache.commons.dbcp.BasicDataSource"
         destroy-method = "close" >
         < property  name = "driverClassName"  value = "oracle.jdbc.driver.OracleDriver"  />
         < property  name = "url"  value = "${db-url}"  />
         < property  name = "username"  value = "${db-username}"  />
         < property  name = "password"  value = "${db-password}"  />
         < property  name = "defaultAutoCommit"  value = "false"  />
         < property  name = "initialSize"  value = "2"  />
         < property  name = "maxActive"  value = "10"  />
         < property  name = "maxWait"  value = "60000"  />
     </ bean >
     
     < bean  id = "${db-source-jndi}"  class = "org.springframework.jndi.JndiObjectFactoryBean" >
         < property  name = "jndiName"  value = "appDS"  />        
     </ bean >

即用占位符来代替bean的id,然后在pom.xml里类似下面设置

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
< profile >
             <!-- 本机环境 -->
             < id >local</ id >
             < properties >
                 ...
                 < db-source-jdbc >dataSource</ db-source-jdbc >
                 < db-source-jndi >NONE</ db-source-jndi >
                 < db-url >jdbc:oracle:thin:@172.21.129.51:1521:orcl</ db-url >                
                 < db-username >mu_fsu</ db-username >
                 < db-password >mu_fsu</ db-password >
                 ...
             </ properties >
             <!-- 默认激活本环境 -->
             < activation >
                 < activeByDefault >true</ activeByDefault >
             </ activation >
         </ profile >        
         < profile >
             <!-- 生产环境 -->
             < id >pro</ id >
             < properties >
                 ...
                 < db-source-jdbc >NONE</ db-source-jdbc >
                 < db-source-jndi >dataSource</ db-source-jndi >
                 ...
             </ properties >
         </ profile >
</ profiles >

这样,mvn clean package -P local打包本地开发环境时,将生成

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
< bean  id = "dataSource"  class = "org.apache.commons.dbcp.BasicDataSource"
         destroy-method = "close" >
         < property  name = "driverClassName"  value = "oracle.jdbc.driver.OracleDriver"  />
         < property  name = "url"  value = "jdbc:oracle:thin:@172.21.129.***:1521:orcl"  />
         < property  name = "username"  value = "***"  />
         < property  name = "password"  value = "***"  />
         < property  name = "defaultAutoCommit"  value = "false"  />
         < property  name = "initialSize"  value = "2"  />
         < property  name = "maxActive"  value = "10"  />
         < property  name = "maxWait"  value = "60000"  />
     </ bean >
     
     < bean  id = "NONE"  class = "org.springframework.jndi.JndiObjectFactoryBean" >
         < property  name = "jndiName"  value = "appDS"  />        
     </ bean >

而打包生产环境 mvn clean package -P pro时,生成

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
< bean  id = "NONE"  class = "org.apache.commons.dbcp.BasicDataSource"
         destroy-method = "close" >
         < property  name = "driverClassName"  value = "oracle.jdbc.driver.OracleDriver"  />
         < property  name = "url"  value = "${db-url}"  />
         < property  name = "username"  value = "${db-username}"  />
         < property  name = "password"  value = "${db-password}"  />
         < property  name = "defaultAutoCommit"  value = "false"  />
         < property  name = "initialSize"  value = "2"  />
         < property  name = "maxActive"  value = "10"  />
         < property  name = "maxWait"  value = "60000"  />
     </ bean >
     
     < bean  id = "dataSource"  class = "org.springframework.jndi.JndiObjectFactoryBean" >
         < property  name = "jndiName"  value = "appDS"  />        
     </ bean >

spring配置的其它跟数据库相关的bean,约定引用dataSource这个名称的bean即可

 

2、不同webserver环境,依赖jar包,是否打包的问题

weblogic上,允许多个app,把共用的jar包按约定打包成一个war文件,以library的方式部署,然后各应用在WEB-INF/weblogic.xml中,用类似下面的形式

?
1
2
3
4
5
6
7
<? xml  version = "1.0"  encoding = "utf-8" ?>
< weblogic-web-app  xmlns = "http://www.bea.com/ns/weblogic/90" >
     ...
     < library-ref >
         < library-name >my-share-lib</ library-name >
     </ library-ref >
</ weblogic-web-app >

指定共享library 的名称即可。这样的好处是,即节省了服务器开销,而且各app打包时,就不必再重复打包这些jar文件,打包后的体积大大减少,上传起来会快很多。

而其它webserver上却未必有这个机制,一般为了方便,我们开发时,往往采用一些轻量级的webserver,比如:tomcat,jetty,jboss 之类,正式部署时才发布到weblogic下,这样带来的问题就是,本机打包时,要求这些依赖jar包,全打包到app的WEB-INF/lib下;而生产环境下,各应用的WEB-INF/lib下并不需要这些jar文件,同样还是用profile来搞定,先处理pom.xml,把依赖项改成类似下面的形式:

?
1
2
3
4
5
6
< dependency >
             < groupId >dom4j</ groupId >
             < artifactId >dom4j</ artifactId >
             < version >1.6.1</ version >
             < scope >${jar.scope}</ scope >
         </ dependency >

即scope这里,用一个占位符来代替,然后profile这样配置

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
< profile >
             <!-- 本机环境 -->
             < id >local</ id >
             < properties >
                 < jar.scope >compile</ jar.scope >
                 ...
             </ properties >
             <!-- 默认激活本环境 -->
             < activation >
                 < activeByDefault >true</ activeByDefault >
             </ activation >
         </ profile >
         < profile >
             <!-- 生产环境 -->
             < id >pro</ id >
             < properties >
                 < jar.scope >provided</ jar.scope >
                 ...
             </ properties >
         </ profile >

在maven里,如果一个依赖项的scope是provided,表示由容器提供,打包时将不会打包进最终的package里,所以这样配置后,生产环境打包时,依赖项的scope全变成了provided,即不打包进war文件,而本机环境下,因为scope是compile,所以会打包到war里


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值