背景
项目构建的时候,需要根据不同的场景来改变项目中的属性资源,最为常见的莫过于数据库连接配置了,试想有生产环境、测试缓存、发布环境等,需要为不同的场景下来动态的改变数据库的连接配置。而使用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>