Filter叫做拦截器, 对目标资源拦截,拦截HTTP请求和HTTP响应,本质是对url进行拦截。
与serlvet不同的是, Filter的初始化是随着服务器启动而启动。
在Filter接口中定义了三个方法
init 初始化
doFilter 执行过滤【核心方法】
destroy 销毁
Filter的配置和servlet类似,
<filter>
<filter-name>Filter2</filter-name>
<filter-class>cn.itcast.filter.Filter2</filter-class>
<!-- 为过滤器配置初始化参数 -->
<init-param>
<param-name>company</param-name>
<param-value>cc</param-value>
</init-param>
</filter>
<!-- 配置过滤器拦截哪个web 资源 -->
<filter-mapping>
<filter-name>Filter1</filter-name>
<!-- 拦截路径 -->
<url-pattern>/hello.jsp</url-pattern>
<url-pattern>/index.jsp</url-pattern>
<!-- 过滤HelloServlet 通过Servlet名字对Servlet进行过滤 也可以按照url 即servlet的访问路径 /* 拦截所有 -->
<!-- <url-pattern>/hello</url-pattern>-->
<servlet-name>HelloServlet</servlet-name>
<!-- 在转发时进行过滤 如果不配置 默认request-->
<dispatcher>FORWARD</dispatcher>
<dispatcher>REQUEST</dispatcher>
</filter-mapping>
案例 通过serllet对get提交的数据进行编码处理,当然实际开发中,建议使用post方式提交数据。
当然关于get方式中文乱码,有两种方式解决
第一种 配置tomcat server.xml URIEncoding="utf-8" 【一般不建议这样做】
第二种 new String(xxx.getBytes("ISO-8859-1"),"utf-8");
package cn.demo.filter;
import java.io.IOException;
import java.io.UnsupportedEncodingException;
import java.util.Map;
import java.util.Set;
import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletRequestWrapper;
public class EncodingFilter implements Filter {
@Override
public void destroy() {
}
@Override
public void doFilter(ServletRequest request, ServletResponse response,
FilterChain chain) throws IOException, ServletException {
// 解决post
request.setCharacterEncoding("utf-8");
// 解决get
EncodingRequest encodingRequest = new EncodingRequest(
(HttpServletRequest) request);
chain.doFilter(encodingRequest, response);
}
@Override
public void init(FilterConfig filterConfig) throws ServletException {
}
}
class EncodingRequest extends HttpServletRequestWrapper {
private HttpServletRequest request;
private boolean hasEncode = false;
public EncodingRequest(HttpServletRequest request) {
super(request);
this.request = request;
}
@Override
public String getParameter(String name) {
// 通过getParameterMap方法完成
String[] values = getParameterValues(name);
if (values == null) {
return null;
}
return values[0];
}
@Override
public String[] getParameterValues(String name) {
// 通过getParameterMap方法完成
Map<String, String[]> parameterMap = getParameterMap();
String[] values = parameterMap.get(name);
return values;
}
@Override
public Map getParameterMap() {
Map<String, String[]> parameterMap = request.getParameterMap();
String method = request.getMethod();
if (method.equalsIgnoreCase("post")) {
return parameterMap;
}
// get提交方式 手动转码 , 这里的转码只进行一次 所以通过 hasEncode 布尔类型变量控制
if (!hasEncode) {
Set<String> keys = parameterMap.keySet();
for (String key : keys) {
String[] values = parameterMap.get(key);
if (values == null) {
continue;
}
for (int i = 0; i < values.length; i++) {
String value = values[i];
// 解决get
try {
value = new String(value.getBytes("ISO-8859-1"),
"utf-8");
// values是一个地址
values[i] = value;
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
}
}
// 一次转码完成后,设置转码状态为true
hasEncode = true;
}
}
return parameterMap;
}
}