JavaWeb 之过滤器

JavaWeb 之过滤器

1. 什么是过滤器

  • Servlet 是用来处理请求的, 过滤器是用来拦截请求的.
  • 当用户请求某个 Servlet 时,会先执行部署在这个请求上的 Filter, 而 Filter 决定是否调用 Servlet.

    当执行 Servlet 代码完成后, 还会执行 Filter 后面的代码!!
  • 它会在一组资源(jsp, servlet, css, html 等等)的前面执行.
  • Filter 是单例的!!

2. 编写过滤器

2.1 步骤

  • 写一个类实现 Filter 接口;
  • 在 web.xml 中进行配置.

2.2 filter 接口的三个方法

  1. void init(FilterConfig)
    • 创建之后,马上执行, Filter 会在服务器启动时创建!
  2. void destroy()
    • 销毁之前执行. 在服务器关闭时,销毁!
  3. void doFilter(ServletRequest, ServletResponse, FilterChain)
    • 每次过滤时,都会执行.

2.3 web.xml 中的配置

<filter>
<filter-name>xxx</filter-name>
<filter-class>cn.itcast.web.filter.AFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>xxx</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>

2.4 Filter 相关的类型

  1. FilterConfig, 与 ServletConfig 相似

    • 获取初始化参数: getInitParameter();
    • 获取过滤器名称: getFilterName();
    • 获取 application: getServletContext(); (较常用)
    • 获取所有初始化参数的名称: Enumeration getInitParameterNames();
  2. FilterChain

    • doFilter(ServletRequest, ServletResponse), 表示放行!

      相当于调用了目标 Servlet 的 service() 方法.

2.5 多过滤器

  • FilterChain 的 doFilter() 方法: 执行目标资源, 或是执行下一个过滤器!
// 如果访问 AServlet, 需要经过AFilter 和 BFilter 两个过滤器,
AFilter#start...
BFilter#start...
AServlet...
BFilter#end...
AFilter#end...

3. 过滤器的四种拦截方式

  • 拦截请求(默认拦截方式)
  • 拦截转发
  • 拦截包含
  • 拦截错误
// <filter-mapping> 进行配置 <dispatcher> 元素

    <filter>
<filter-name>AFilter</filter-name>
<filter-class>cn.itcast.web.filter.AFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>AFilter</filter-name>
<url-pattern>/AServlet</url-pattern>
<dispatcher>REQUEST</dispatcher> // 拦截请求
<dispatcher>FORWARD</dispatcher> // 拦截转发
<dispatcher>INCLUDE</dispatcher> // 拦截包含
<dispatcher>ERROR</dispatcher> // 拦截错误
</filter-mapping>

4. 多个过滤器的执行顺序

  • <filter-mapping> 的配置顺序决定了过滤器的执行顺序!!

5. 过滤器的应用场景

  1. 执行目标资源之前做预处理工作, 例如设置编码,这种通常都会放行, 只是在目标资源执行之前做一些准备工作;
  2. 通过条件判断是否放行, 例如校验当前用户是否已经登录, 或者用户 IP 是否已经被禁用;
  3. 在目标资源执行后, 做一些后续的特殊处理工作, 例如对目标资源输出的数据进行处理;
// 示例一: 分 IP 统计访问次数
/*
* 分析:
* 1. 使用 Map<String, Integer> 来装载统计的数据;
* 2. 使用 ServletContextListener, 在服务器启动时完成创建;
* 3. Map 保存到 ServletContext 中;
* 因为 Map 需要在 Filter 中用来保存数据,
* 而页面需要打印 Map 中的数据.
*/ // AListener
public class AListener implements SerlvetContextListener { // 在服务器启动时, 创建Map, 保存到 ServletContext中
public void contextInitialized(SerlvetContextEvent sce){ Map<String, Integer> map = new LinkedHashMap<String,Integer>(); // 得到 ServletContext
ServletContext application = sce.getServletContext(); // 把 map 保存到 application 中
application.setAttribute("map",map);
} public void contextDestroyed(ServletContextEvent sce){ }
} // AFilter public class AFilter implements Filter{
private FilterConfig config; public void destory(){ } public void init(FilterConfig config) throws ServletException{
// 赋值
this.config = config;
} public void doFilter(ServletRequest request, ServletResponse response,
FilterChain chain) throws IOException, ServletException{ // 得到 application 中的 map
ServletContext app = config.getServletContext(); Map<String,Integer> map = (Map<String,Integer>)app.getAttribute("map"); // 从 request 域中得到 ip 地址
String ip = request.getRemoteAddr(); // 查看 map 中是否存在这个 ip 对应的访问次数, 如果存在, 把次数加 1 再保存回去
// 如果不存在这个 ip, 那么设置这个 ip 的访问次数为 1 if(map.containsKey(ip)){ int cnt = map.get(ip);
map.put(ip,cnt+1); }else{
map.put(ip,1);
} // 把 map 放回到 application 中
app.setAttribute("map",map); // 放行
chain.doFilter(request,response);
}
} // 示例二: 解决全站字符乱码(POST 和 GET 中文乱码问题) // index.jsp
<body>
<h1>主页</h1> <%-- POST 请求 --%>
<form action="<c:url value='/AServlet'/>" method="post">
用户名:<input type="text" name="username" value="张三"/><br/>
<input type="submit" value="登录"/>
</form> <%-- GET 请求, tomcat 8.0 以上版本,没有乱码问题 --%>
<a href="<c:url value='/AServlet?username=李四'"/>点击这里</a>
</body> // EncodingFilter // 处理 POST 请求编码问题
request.setCharacterEncoding("utf-8"); // 处理 GET 请求编码问题
// 需要调包 request:
// 写一个 request 的装饰类
// 在放行时,使用我们自己的 request if(req.getMethod().equals("GET")){
HttpServletRequest req = (HttpServletRequest)request; EncodingRequest er = new EncodingRequest(req); chain.doFilter(er,response);
}else if(req.getMethod().equals("POST")){
chain.doFilter(request,response);
} // EncodingRequest 类, 即 request 的装饰类
// 装饰 request 的 getParameter(String name) 方法
public class EncodingRequest implements HttpServletRequest{ private HttpServletRequest request; // 有参构造方法(是你,还有你)
public EncodingRequest(HttpServletRequest request){
this.request = request;
} // 增强 request 的方法
public String getParameter(String name){
String value = request.getParameter(name); // 处理编码问题
try{
value = new String(value.getBytes("iso-8859-1"),"utf-8");
}catch(Exception e){
throw new RuntimeException(e);
}
return value;
} // 复写 request 其他方法 (一切拜托你)
....
} // EncodingRequest 类的升级版
// EncodingRequest 类继承 HttpServletRequestWrapper 类即可
// HttpServletRequestWrapper 实现了 HttpServletRequest 中的所有方法 public class EncodingRequest extends HttpServletRequestWrapper{
private HttpServletRequest req; // 构造方法
public EncodingRequest(HttpServletRequest request){
// 将 request 参数传递给父类 HtttpServletRequestWrapper
super(request);
this.req = request;
} // 需要增强的方法
public String getParameter(String name){
String value = req.getParameter(name);
try{
value = new String(value.getBytes("iso-8859-1"),"utf-8");
}catch(UnsupportedEncodingException e){
throw new RuntimeException(e);
}
return value;
}
}

参考资料:

上一篇:TicTacToe井字棋 by reinforcement learning


下一篇:[Web 前端] 解决因inline-block元素导致的空白间距和元素下沉