Maven中profile和filtering实现多个环境下的属性过滤

背景
项目构建的时候,需要根据不同的场景来改变项目中的属性资源,最为常见的莫过于数据库连接配置了,试想有生产环境、测试缓存、发布环境等,需要为不同的场景下来动态的改变数据库的连接配置。而使用maven就可以帮我们解决这些问题。下面就来分享一下maven中的profile和filtering的属性。
为了便于测试一下功能,需要搭建maven的web项目,具体配置请详见如何用maven创建web项目。

filtering功能
主要用来替换项目中的资源文件(*.xml、*.properties)当中的${...},比如${db.url},那么如果配置了db.url=aaa的话,在项目编译的时候,就会自动的把${db.url}替换为aaa,下面以实例来讲解一下。
创建完maven的web项目后,会看到src/main/resources的目录,在此目录下面创建个“test.properties”,里面随便来上一行,例如Hello ${user.name},好了,接下来修改我们的pom文件,来启动filtering功能。

<span style="font-family:Comic Sans MS;font-size:18px;"><project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"  
    xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">  
    <modelVersion>4.0.0</modelVersion>  
    <groupId>testwebProject</groupId>  
    <artifactId>com.test.web.test</artifactId>  
    <version>0.0.1-SNAPSHOT</version>  
    <packaging>war</packaging>  
    <build>  
        <!--第一种方式,两种方式都需要指定需要编译的目录 -->  
        <resources>  
            <resource>  
                <directory>src/main/resources</directory>  
                <!--可以在此配置过滤文件  -->  
                <includes>  
                    <include>**/*.xsd</include>  
                    <include>**/*.properties</include>  
                </includes>  
                <!--开启filtering功能  -->  
                <filtering>true</filtering>  
            </resource>  
        </resources>
  
        <plugins>  
            <plugin>  
                <artifactId>maven-war-plugin</artifactId>  
                <configuration>  
                    <version>2.5</version>  
                </configuration>  
            </plugin>  
        </plugins>  
    </build>  
</project></span>

然后编译我们的maven项目:
$mvn clean compile -Duser.name=tom
$编译完后,查看输出文件 target/classes/test.properties 的内容,可见原先的 “Hello {user.name}” 已经变成 “Hello Tom”。

上面如果麻烦的话,也可以把filtering用到的变量写在项目得属性段里面,如下面的方式:

<!--也可以配置到外部属性里面 -->  
<properties>  
    <user.name>Lucky</user.name>  
    <user.age>50</user.age>  
</properties>

进行编译,$mvn clean compile,在此查看的话,就会看到属性被替换的效果。
当然了,如果属性比较多的话,那么此时可以把属性单独抽取出来一个*.properties文件来保存,例如我们在pom.xml的同级目录下面创建一个project.properties,里面指明我们的内容
user.name=Lucky
紧接着在修改我们的pom文件,如下:

<span style="font-family:Comic Sans MS;font-size:18px;"><project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"  
    xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">  
    <modelVersion>4.0.0</modelVersion>  
    <groupId>testwebProject</groupId>  
    <artifactId>com.test.web.test</artifactId>  
    <version>0.0.1-SNAPSHOT</version>  
    <packaging>war</packaging>  
  
    <build>  
        <!--第一种方式,两种方式都需要指定需要编译的目录 -->  
        <resources>  
            <resource>  
                <directory>src/main/resources</directory>  
                <!--可以在此配置过滤文件 -->  
                <includes>  
                    <include>**/*.xsd</include>  
                    <include>**/*.properties</include>  
                </includes>  
                <!--开启filtering功能 -->  
                <filtering>true</filtering>  
            </resource>  
        </resources>  
  
        <filters>  
            <!-- 是以该pom文件路径作为参考 -->  
            <filter>project.properties</filter>  
        </filters>
  
        <plugins>  
            <plugin>  
                <artifactId>maven-war-plugin</artifactId>  
                <configuration>  
                    <version>2.5</version>  
                </configuration>  
            </plugin>  
        </plugins>  
    </build>  
</project>
</span>

再次执行编译命令的话,就会看到效果。

profile功能
允许在项目文件(pom.xml)里面定义若干个profile段,然后在编译时选择其中的一个用于覆盖项目文件原先的定义。接着上一个例子,如果我们需要为开发环境和生产环境定义不同的 user.name 属性值,则我们在项目目录里创建两个属性文件分别是pre.properties和dev.properties,然后再每个文件里分别写入user.name=lucky和user.name=wangwang,然后在此修改我们的pom文件,修改后如下所示:

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
    xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> 
    <modelVersion>4.0.0</modelVersion> 
    <groupId>testwebProject</groupId> 
    <artifactId>com.test.web.test</artifactId> 
    <version>0.0.1-SNAPSHOT</version> 
    <packaging>war</packaging> 
   
    <build> 
        <!--第一种方式,两种方式都需要指定需要编译的目录 --> 
        <resources> 
            <resource> 
                <directory>src/main/resources</directory> 
                <!--可以在此配置过滤文件 --> 
                <includes> 
                    <include>**/*.xsd</include> 
                    <include>**/*.properties</include> 
                </includes> 
                <!--开启filtering功能 --> 
                <filtering>true</filtering> 
            </resource> 
        </resources> 
   
        <profiles> 
            <profile> 
                <id>dev</id> 
                <activation> 
                <!--默认的编译选项  --> 
                    <activeByDefault>true</activeByDefault> 
                </activation> 
                <build> 
                    <filters> 
                        <filter>dev.properties</filter> 
                    </filters> 
                </build> 
            </profile> 
   
            <profile>
                <id>pre</id> 
                <build> 
                    <filters> 
                        <filter>pre.properties</filter> 
                    </filters> 
                </build> 
            </profile> 
        </profiles>
   
        <plugins> 
            <plugin> 
                <artifactId>maven-war-plugin</artifactId> 
                <configuration> 
                    <version>2.5</version> 
                </configuration> 
            </plugin> 
        </plugins> 
    </build> 
</project>

在编译项目时,可以使用 -P 参数指定需要使用的 profile 的 id,比如下面命令将会使用 dev profile:

$mvn clean compile -Pdev
如果想使用pre,只需要改为以下即可
$mvn clean compile -Ppre
假如不指定 -P 参数的话,则会使用 activeByDefault=true 的一项(即 dev)。

补充(添加filtering导致二进制文件异常问题)

最近在项目中使用bootstrap时,发现页面中图标显示异常,通过F12调试发现以下告警:

OTS parsing error: Failed to convert WOFF 2.0 font to SFNT
index:1 OTS parsing error: incorrect file size in WOFF header
index:1 OTS parsing error: incorrect entrySelector for table directory
main:1 OTS parsing error: Failed to convert WOFF 2.0 font to SFNT
main:1 OTS parsing error: incorrect file size in WOFF header
main:1 OTS parsing error: incorrect entrySelector for table directory

发现是因为分环境打包,开启filtering功能后,会破坏有二进制内容的文件。 因为项目中的图标引用了一个字库,该字库为二进制文件,打包后被破坏。所以这里我将需要过滤的文件单独列出来,然后其他文件配置成无需过滤。 

<resources>
    <resource>
        <directory>src/main/resources</directory>
        <filtering>true</filtering>
        <includes>
            <include>application.yml</include>
            <include>app.properties</include>
        </includes>
    </resource>
    <!--以下这个配置很重要,如果不加这个配置,上面2个文件以外的其他文件将不会生成到war中  -->
    <resource>
        <directory>src/main/resources</directory>
        <filtering>false</filtering>
        <excludes>
            <exclude>application.yml</exclude>
            <exclude>app.properties</exclude>
        </excludes>
    </resource>
</resources>
原文地址:https://www.cnblogs.com/atai/p/6933966.html