菜鸟教程  传送门

  

  过滤器Filter::JavaWeb三大组件之一,它与Servlet很相似,过滤器是用来拦截请求的,而不是处理请求的

  当用户请求某个Servlet时,会先执行部署在这个请求上的Filter,如果Filter“放行”,那么会继承执行用户请求的Servlet;如果Filter不“放行”,那么就不会执行用户请求的Servlet。

过滤器Filter生命周期

package com.Gary.filter;

import java.io.IOException;
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.annotation.WebFilter; @WebFilter("/*")
public class GaryFilter implements Filter { public GaryFilter() {
System.out.println("过滤器-构造方法");
} public void destroy() {
System.out.println("过滤器-destroy()");
} public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
chain.doFilter(request, response);
System.out.println("过滤器-doFilter()");
} public void init(FilterConfig fConfig) throws ServletException {
System.out.println("过滤器-init()");
} }

GaryFilter.java

  init(FilterConfig):在服务器启动时会创建Filter实例,并且每个类型的Filter只创建一个实例,从此不再创建!在创建完Filter实例后,会马上调用init()方法完成初始化工作,这个方法只会被执行一次;

   doFilter(ServletRequest req,ServletResponse res,FilterChain chain):这个方法会在用户每次访问“目标资源(<url->pattern>index.jsp</url-pattern>)”时执行,如果需要“放行”,那么需要调用FilterChain的doFilter(ServletRequest,ServletResponse)方法,如果不调用FilterChain的doFilter()方法,那么目标资源将无法执行;

  destroy():服务器会在创建Filter对象之后,把Filter放到缓存中一直使用,通常不会销毁它。一般会在服务器关闭时销毁Filter对象,在销毁Filter对象之前,服务器会调用Filter对象的destory()方法。

  @WebFilter("/*")表示过滤全部的请求

  @WebFilter("/login.jsp")表示过滤登录的请求

  每发起一个请求过滤器都会调用一次doFilter()【通常都是对请求进行过滤,对请求过滤放到chain.doFilter(request, response)上边】

    public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
System.out.println("过滤器-请求前");
chain.doFilter(request, response);
System.out.println("过滤器-请求后");
}

  当过滤登录请求时可先指定路径@WebFilter("/login.jsp*")后,修改doFilter()方法,用户访问login.jsp时页面重定向到index.jsp

    public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
chain.doFilter(request, response);
//当访问login.jsp时页面请求重定向
((HttpServletResponse)response).sendRedirect("index.jsp");
}

  通过Web.xml中配置

<!--   注册    告诉web有哪些filter并告诉其路径 -->
<filter>
<filter-name>xxx</filter-name>
<filter-class>com.Gary.filter.GaryFilter</filter-class>
</filter> <!-- filter映射 -->
<filter-mapping>
<filter-name>xxx</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
<filter>指定一个过滤器。
<filter-name>用于为过滤器指定一个名字,该元素的内容不能为空。
<filter-class>元素用于指定过滤器的完整的限定类名。
<init-param>元素用于为过滤器指定初始化参数,它的子元素<param-name>指定参数的名字,<param-value>指定参数的值。
在过滤器中,可以使用FilterConfig接口对象来访问初始化参数。
<filter-mapping>元素用于设置一个 Filter 所负责拦截的资源。一个Filter拦截的资源可通过两种方式来指定:Servlet 名称和资源访问的请求路径
<filter-name>子元素用于设置filter的注册名称。该值必须是在<filter>元素中声明过的过滤器的名字
<url-pattern>设置 filter 所拦截的请求路径(过滤器关联的URL样式)
<servlet-name>指定过滤器所拦截的Servlet名称。
<dispatcher>指定过滤器所拦截的资源被 Servlet 容器调用的方式,可以是REQUEST,INCLUDE,FORWARD和ERROR之一,默认REQUEST。用户可以设置多个<dispatcher>子元素用来指定 Filter 对资源的多种调用方式进行拦截。
<dispatcher>子元素可以设置的值及其意义
REQUEST:当用户直接访问页面时,Web容器将会调用过滤器。如果目标资源是通过RequestDispatcher的include()或forward()方法访问时,那么该过滤器就不会被调用。
INCLUDE:如果目标资源是通过RequestDispatcher的include()方法访问时,那么该过滤器将被调用。除此之外,该过滤器不会被调用。
FORWARD:如果目标资源是通过RequestDispatcher的forward()方法访问时,那么该过滤器将被调用,除此之外,该过滤器不会被调用。
ERROR:如果目标资源是通过声明式异常处理机制调用时,那么该过滤器将被调用。除此之外,过滤器不会被调用。

web.xml配置各节点说明

过滤器的应用场景:

  (一)执行目标资源之前做预处理工作,例如设置编码,这种试通常都会放行,只是在目标资源执行之前做一些准备工作;

  (二)通过条件判断是否放行,例如校验当前用户是否已经登录,或者用户IP是否已经被禁用;

  (三)在目标资源执行后,做一些后续的特殊处理工作,例如把目标资源输出的数据进行处理;

  (一)【静态】当编码通过静态设置时可直接放到过滤器doFilter()中

//    HttpServletRequest ServletRequest
// HttpServletRequest ServletResponse
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
//对编码进行过滤 类型为ServletResponse
request.setCharacterEncoding("utf-8");
chain.doFilter(request, response);
}

  (一)【动态】当编码通过动态设置时需放到Web.xml中

<!--   注册    告诉web有哪些filter并告诉其路径 -->
<filter>
<filter-name>xxx</filter-name>
<filter-class>com.Gary.filter.EncodeFilter</filter-class>
<init-param>
<param-name>Encoding</param-name>
<param-value>UTF-8</param-value>
</init-param>
</filter> <!-- filter映射 -->
<filter-mapping>
<filter-name>xxx</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>

  (一)过滤器中init()初始化编码类型

    private String encoding;

    public void init(FilterConfig fConfig) throws ServletException {
encoding = fConfig.getInitParameter("Encoding");
}

  (二)doFilter()中设置权限管理过滤  完整项目代码

  Web,xml中配置过滤器映射

  <filter>
<filter-name>AdminFilter</filter-name>
<filter-class>com.Gary.filter.AdminFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>AdminFilter</filter-name>
<!-- 访问admin目录下文件做权限过滤 -->
<url-pattern>/admin/*</url-pattern>
</filter-mapping>

  实现过滤器中的doFilter,isAdmin对用户是否是管理员进行判断,管理员返回值为True

public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {

        HttpServletRequest httpReq = (HttpServletRequest)request;
HttpServletResponse httpResp = (HttpServletResponse) response;
Object o = httpReq.getSession().getAttribute("user");
if(o==null) {
httpResp.sendRedirect(httpReq.getContextPath()+"/index.jsp");
}else {
User u = (User)o;
if(u.isAdmin()) {
// 对权限进行放行
chain.doFilter(request, response);
}else {
// 重定向回首页
httpResp.sendRedirect(httpReq.getContextPath()+"/index.jsp");
}
}
}
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://xmlns.jcp.org/xml/ns/javaee" xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd" id="WebApp_ID" version="3.1">
<display-name>Gary04</display-name>
<welcome-file-list>
<welcome-file>index.html</welcome-file>
<welcome-file>index.htm</welcome-file>
<welcome-file>index.jsp</welcome-file>
<welcome-file>default.html</welcome-file>
<welcome-file>default.htm</welcome-file>
<welcome-file>default.jsp</welcome-file>
</welcome-file-list>
<filter>
<filter-name>AdminFilter</filter-name>
<filter-class>com.Gary.filter.AdminFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>AdminFilter</filter-name>
<!-- 访问admin目录下文件做权限过滤 -->
<url-pattern>/admin/*</url-pattern>
</filter-mapping>
</web-app>

web.xml

package com.Gary.filter;

import java.io.IOException;
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.annotation.WebFilter;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse; import com.Gary.model.User; //@WebFilter("/AdminFilter")
public class AdminFilter implements Filter { public AdminFilter() {
// TODO Auto-generated constructor stub
} public void destroy() {
// TODO Auto-generated method stub
} public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException { HttpServletRequest httpReq = (HttpServletRequest)request;
HttpServletResponse httpResp = (HttpServletResponse) response;
Object o = httpReq.getSession().getAttribute("user");
if(o==null) {
httpResp.sendRedirect(httpReq.getContextPath()+"/index.jsp");
}else {
User u = (User)o;
if(u.isAdmin()) {
// 对权限进行放行
chain.doFilter(request, response);
}else {
// 重定向回首页
httpResp.sendRedirect(httpReq.getContextPath()+"/index.jsp");
}
}
} /**
* @see Filter#init(FilterConfig)
*/
public void init(FilterConfig fConfig) throws ServletException {
// TODO Auto-generated method stub
} }

AdminFilter.java

总结

Filter的三个方法

  void init(FilterConfig):在Tomcat启动时被调用;

  void destroy():在Tomcat关闭时被调用;

  void doFilter(ServletRequest,ServletResponse,FilterChain):每次有请求时都调用该方法;

FilterConfig类:与ServletConfig相似,用来获取Filter的初始化参数

  ServletContext getServletContext():获取ServletContext的方法;

  String getFilterName():获取Filter的配置名称;

  String getInitParameter(String name):获取Filter的初始化配置,与<init-param>元素对应;

  Enumeration getInitParameterNames():获取所有初始化参数的名称。

FilterChain类

  void doFilter(ServletRequest,ServletResponse):放行!表示执行下一个过滤器,或者执行目标资源。可以在调用FilterChain的doFilter()方法的前后添加语句,在FilterChain的doFilter()方法之前的语句会在目标资源执行之前执行,在FilterChain的doFilter()方法之后的语句会在目标资源执行之后执行

各拦截方式:REQUEST、FORWARD、INCLUDE、ERROR,默认是REQUEST方式

  REQUEST:拦截直接请求方式

  FORWARD:拦截请求转发方式

  INCLUDE:拦截请求包含方式

  ERROR:拦截错误转发方式

  <filter>
<filter-name>DispatcherFilter</filter-name>
<filter-class>com.Gary.filter.DispatcherFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>DispatcherFilter</filter-name>
<url-pattern>/dispatcher1.jsp</url-pattern>
<dispatcher>REQUEST</dispatcher>
</filter-mapping>

最新文章

  1. thinkphp使用ajax
  2. configure Git to accept a particular self-signed server certificate for a particular https remote
  3. python接口测试wsdl
  4. Spark standlone安装与配置
  5. outlook无法创建保存附件解决
  6. 超链接点击后不显示hover
  7. POJ 1850 Code 字符串 难度:1
  8. 翻译:wiki中的business logic词条
  9. CSS 样式二
  10. PL/SQL — 隐式游标
  11. bzoj1060 [ZJOI2007]时态同步
  12. boost锁的概述
  13. angualr4 路由 总结笔记
  14. python安装第三方库
  15. Servlet之Session处理
  16. JUnit 异常处理
  17. [RHEL 7]ISCSI服务端及客户端连接配置
  18. 基于jeesite的cms系统(五):wangEditor富文本编辑器
  19. HanLP用户自定义词典源码分析
  20. Asp.net MVC5 学习笔记

热门文章

  1. P3376 网络流-最大流模板题(Dinic+当前弧优化)
  2. 删除链表中重复的结点——牛客剑指offer
  3. deepin 15.10.1 GTX1060 NVIDIA 驱动安装,双屏显示问题记录
  4. 13 UA池和代理池
  5. 解决Idea、WebStorm下使用Vue cli脚手架项目无法使用Webpack别名的问题
  6. Vue初始化一个项目
  7. se37 函数中的异常使用
  8. Clang调试deadcode思路
  9. css3中单位rem与.less结合布局
  10. 能ping通某网页,但无法访问网页的处理