通過properties檔案配置web.xml中的引數
前言
因為公司專案需要,目前有本地環境、測試環境、開發環境。每次在將專案打包成war包的時候,都需要修改多處的配置,而使用maven的profile打包專案的時候,可以根據執行打包命令時所帶的引數來進行自動修改。
但是這種方式只對properties檔案生效,即可以自動修改properties中的引數,但是公司的專案有一個web.xml中的配置引數也需要修改,這時候就要考慮如何通過properties檔案動態修改web.xml中的引數了。
實現思路
web.xml中需要修改的部分:
<web-app xmlns="http://java.sun.com/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee
http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd"
version="3.0" metadata-complete="true">
<!--用maven建立的web-app需要修改servlet的版本為3.1 -->
<welcome-file-list>
<welcome-file >/index.jsp</welcome-file>
</welcome-file-list>
<!--配置DispatcherServlet -->
<servlet>
<servlet-name>mypage-dispatcher</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<!-- 配置SpringMVC 需要配置的檔案 spring-dao.xml,spring-service.xml,spring-web.xml
Mybites -> spring -> springMvc -->
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:spring/spring-*.xml</param-value>
</init-param>
</servlet>
<servlet-mapping>
<servlet-name>mypage-dispatcher</servlet-name>
<!--預設匹配所有請求 -->
<url-pattern>/</url-pattern>
</servlet-mapping>
<servlet-mapping>
<servlet-name>default</servlet-name>
<url-pattern>*.html</url-pattern>
</servlet-mapping>
<filter>
<filter-name>testFilter</filter-name>
<filter-class>com.solr.filter.StringFilter</filter-class>
<init-param>
<param-name>jersey.config.server.provider.packages</param-name>
<param-value>
com.sgm.tac.tid.common.action;
</param-value>
</init-param>
</filter>
<filter-mapping>
<filter-name>testFilter</filter-name>
<url-pattern>*.*</url-pattern>
</filter-mapping>
</web-app>
這裡需要改動的是45行,這是過濾器的初始化引數。不同的環境這裡的引數是不一樣的,開始的思路是能否像application.xml中載入的properties檔案一樣,通過${username}這種方式獲取properties中username對應的value。但是後來發現在web.xml中貌似是不好實現的。
在這樣的需求下,web.xml就需要以編碼的方式來實現配置。spring4.0以上的版本支援web.xml的編碼配置。實現AbstractAnnotationConfigDispatcherServletInitializer介面,在servlet3.0中web.xml啟動時會檢測該介面實現類,從能夠在實現類中去配置filter。
需要注意的是以上的實現,依賴servlet-api-3.0.jar和spring-webmvc-4.0以上版本jar包。
配置web.xml的類
package com.solr.filter;
import java.util.EnumSet;
import javax.servlet.DispatcherType;
import javax.servlet.Filter;
import javax.servlet.FilterRegistration;
import javax.servlet.FilterRegistration.Dynamic;
import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import org.springframework.web.servlet.support.AbstractAnnotationConfigDispatcherServletInitializer;
import com.solr.util.PropUtils;
public class WebAppInitializer extends AbstractAnnotationConfigDispatcherServletInitializer {
@Override
protected Class<?>[] getRootConfigClasses() {
// TODO Auto-generated method stub
return null;
}
@Override
protected Class<?>[] getServletConfigClasses() {
// TODO Auto-generated method stub
return null;
}
@Override
protected String[] getServletMappings() {
// TODO Auto-generated method stub
return null;
}
@Override
public void onStartup(ServletContext servletContext) throws ServletException {
// 系統啟動時註冊filter
FilterRegistration testFilter = servletContext.addFilter("testFilter", StringFilter.class);
// 設定init param, param可以從properties檔案中讀取或其他方式獲取,提供一個想法
testFilter.setInitParameter("jersey.config.server.provider.packages", PropUtils.getValueByKey("FILTER.NAME"));
testFilter.addMappingForUrlPatterns(EnumSet.allOf(DispatcherType.class) , true, "*.*");
super.onStartup(servletContext);
}
@Override
protected Dynamic registerServletFilter(ServletContext servletContext, Filter filter) {
// TODO Auto-generated method stub
return super.registerServletFilter(servletContext, filter);
}
}
在繼承AbstractAnnotationConfigDispatcherServletInitializer的時候onStartup和registerServletFilter兩個方法沒有自動新增進來,需要自己手動override。
其中onStartup在伺服器啟動的時候會根據配置修改web.xml,此處通過addFilter添加了一個叫做testFilter的過濾器,通過setInitParameter向過濾器中設定引數。而這裡PropUtils是自己寫的一個讀取properties檔案的工具類,這樣就實現了將properties中的值動態新增到web.xml中了,最後打包修改properties的時候就可以修改web.xml了。
filter.properties檔案
FILTER.NAME=HHH
PropUtils工具類
此工具類使用ResourceBundle讀取properties檔案,此工具類是java.util中的方法,其中還有一些stringUtils的方法,用來判斷字串是否為空,將字串轉換成大寫等功能,就不寫在上面了。
package com.solr.util;
import java.text.MessageFormat;
import java.util.Locale;
import java.util.ResourceBundle;
import org.apache.log4j.Logger;
public class PropUtils {
private static final String URL_RESOURCE_FILE_PATH = "props/filter";
private static final Logger LOG = Logger.getLogger(PropUtils.class);
private static final ResourceBundle rb = ResourceBundle.getBundle(URL_RESOURCE_FILE_PATH, Locale.getDefault(),PropUtils.class.getClassLoader());
/**
* @param key 對應properties內的key
* @return properties對應字串
*/
public static String getValueByKey(String key){
return getValueByKey(key,null);
}
/**
* @param key 對應properties內的key
* @param param 引數下標0開始依次排列
* @return properties內填入對應引數的字串
*/
public static String getValueByKey(String key,Object [] param){
String value = "";
try {
value = rb.getString(StringUtils.toUpper(key));
} catch (Exception e) {
LOG.info(e,e);
}
if (StringUtils.isBlank(value)){
return key;
}
String strReturn = "";
if (param == null || param.length == 0){
strReturn = MessageFormat.format(value, param);
}else {
strReturn = value;
}
return strReturn.trim();
}
}
檢視web.xml引數
在啟動伺服器的時候,會對過濾器進行初始化,我們可以在初始化的時候檢視剛才配置的web.xml是否成功。
package com.solr.filter;
import java.io.IOException;
import java.util.Enumeration;
import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
public class StringFilter implements Filter {
@Override
public void init(FilterConfig filterConfig) throws ServletException {
// TODO Auto-generated method stub
System.out.println("init");
Enumeration<String> initParameterNames = filterConfig.getInitParameterNames();
while (initParameterNames.hasMoreElements()) {
String param = (String) initParameterNames.nextElement();
System.out.println(param + ":" + filterConfig.getInitParameter(param));
}
}
@Override
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain)
throws IOException, ServletException {
// TODO Auto-generated method stub
System.out.println("dofilter");
}
@Override
public void destroy() {
// TODO Auto-generated method stub
System.out.println("destroy");
}
}
啟動伺服器進行測試
啟動伺服器的時候報錯了:
八月 17, 2018 2:48:27 下午 org.apache.catalina.core.ContainerBase startInternal
嚴重: A child container failed during start
java.util.concurrent.ExecutionException: org.apache.catalina.LifecycleException: Failed to start component [StandardEngine[Catalina].StandardHost[localhost]]
at java.util.concurrent.FutureTask.report(FutureTask.java:122)
at java.util.concurrent.FutureTask.get(FutureTask.java:192)
at org.apache.catalina.core.ContainerBase.startInternal(ContainerBase.java:1241)
at org.apache.catalina.core.StandardEngine.startInternal(StandardEngine.java:300)
at org.apache.catalina.util.LifecycleBase.start(LifecycleBase.java:145)
at org.apache.catalina.core.StandardService.startInternal(StandardService.java:444)
at org.apache.catalina.util.LifecycleBase.start(LifecycleBase.java:145)
at org.apache.catalina.core.StandardServer.startInternal(StandardServer.java:758)
at org.apache.catalina.util.LifecycleBase.start(LifecycleBase.java:145)
at org.apache.catalina.startup.Catalina.start(Catalina.java:705)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:498)
at org.apache.catalina.startup.Bootstrap.start(Bootstrap.java:294)
at org.apache.catalina.startup.Bootstrap.main(Bootstrap.java:428)
Caused by: org.apache.catalina.LifecycleException: Failed to start component [StandardEngine[Catalina].StandardHost[localhost]]
at org.apache.catalina.util.LifecycleBase.start(LifecycleBase.java:162)
at org.apache.catalina.core.ContainerBase$StartChild.call(ContainerBase.java:1702)
at org.apache.catalina.core.ContainerBase$StartChild.call(ContainerBase.java:1692)
at java.util.concurrent.FutureTask.run(FutureTask.java:266)
at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624)
at java.lang.Thread.run(Thread.java:748)
Caused by: org.apache.catalina.LifecycleException: A child container failed during start
at org.apache.catalina.core.ContainerBase.startInternal(ContainerBase.java:1249)
at org.apache.catalina.core.StandardHost.startInternal(StandardHost.java:819)
at org.apache.catalina.util.LifecycleBase.start(LifecycleBase.java:145)
... 6 more
錯誤的意思大概是載入元件遇到了問題。這個問題是在想通過編碼的方式來實現配置web.xml的時候出現的,即在之前是沒有遇到這個問題的,實現繼承AbstractAnnotationConfigDispatcherServletInitializer,並向web.xml中新增過濾器的時候遇到此問題的。
最終原因是通過編碼新增的過濾器名稱為testFilter,而web.xml中原先就有這個名稱的過濾器,兩個過濾器名稱衝突,造成伺服器啟動失敗。
解決方式:刪除web.xml中原先的過濾器配置,通過編碼新增此過濾器。
web.xml
<web-app xmlns="http://java.sun.com/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee
http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd"
version="3.0" metadata-complete="true">
<!--用maven建立的web-app需要修改servlet的版本為3.1 -->
<welcome-file-list>
<welcome-file>/index.jsp</welcome-file>
</welcome-file-list>
<!--配置DispatcherServlet -->
<servlet>
<servlet-name>mypage-dispatcher</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<!-- 配置SpringMVC 需要配置的檔案 spring-dao.xml,spring-service.xml,spring-web.xml
Mybites -> spring -> springMvc -->
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:spring/spring-*.xml</param-value>
</init-param>
</servlet>
<servlet-mapping>
<servlet-name>mypage-dispatcher</servlet-name>
<!--預設匹配所有請求 -->
<url-pattern>/</url-pattern>
</servlet-mapping>
<servlet-mapping>
<servlet-name>default</servlet-name>
<url-pattern>*.html</url-pattern>
</servlet-mapping>
</web-app>