Servlet 过滤器和异常处理
Servlet 编写过滤器
Servlet 过滤器可以动态地拦截请求和响应,以变换或使用包含在请求或响应中的信息。可以将一个或多个 Servlet 过滤器附加到一个 Servlet 或一组 Servlet。Servlet 过滤器也可以附加到 JavaServer Pages (JSP) 文件和 HTML 页面。调用 Servlet 前调用所有附加的 Servlet 过滤器。
Servlet 过滤器用途
- 在客户端的请求访问后端资源之前,拦截这些请求。
- 在服务器的响应发送回客户端之前,处理这些响应。
使用说明
当 Web 容器启动 Web 应用程序时,它会为您在部署描述符中声明的每一个过滤器创建一个实例。Filter 的执行顺序与在web.xml配置文件中的配置顺序一致,一般把 Filter 配置在所有的 Servlet 之前。
Servlet 过滤器方法
过滤器是一个实现了 javax.servlet.Filter 接口的 Java 类。javax.servlet.Filter 接口定义了三个方法:
public void doFilter (ServletRequest, ServletResponse, FilterChain)
: 该方法完成实际的过滤操作,当客户端请求方法与过滤器设置匹配的URL时,Servlet容器将先调用过滤器的doFilter方法。FilterChain 用户访问后续过滤器public void init(FilterConfig filterConfig)
: web 应用程序启动时,web 服务器将创建Filter 的实例对象,并调用其init方法,读取web.xml配置,完成对象的初始化功能,从而为后续的用户请求作好拦截的准备工作(filter对象只会创建一次,init方法也只会执行一次)。开发人员通过init方法的参数,可获得代表当前 filter 配置信息的 FilterConfig 对象public void destroy()
: Servlet容器在销毁过滤器实例前调用该方法,在该方法中释放Servlet过滤器占用的资源
流程
请求 --> 匹配过滤器 --> doFilter() --> servlet响应前的过滤 --> chain链(servlet响应) --> servlet响应后的过滤
FilterConfig
Filter 的 init 方法中提供了一个 FilterConfig 对象。
注解方式配置 FilterConfig:
1 @WebFilter( 2 filterName = "loginFilter", 3 urlPatterns = "/*", 4 initParams = { 5 @WebInitParam(name = "loginUI", value = "/home/loginUI"), 6 @WebInitParam(name = "loginProcess", value = "home/login"), 7 @WebInitParam(name = "encoding", value = "utf-8") 8 } 9 )
web.xml 文件配置 FilterConfig:
1 <filter> 2 <filter-name>LogFilter</filter-name> 3 <filter-class>com.xxx.test.LogFilter</filter-class> 4 <init-param> 5 <param-name>site</param-name> 6 <param-value>知乎</param-value> 7 </init-param> 8 </filter>
在 init 方法使用 FilterConfig 对象获取参数:
1 public void init(FilterConfig config) throws ServletException { 2 // 获取初始化参数 3 String site = config.getInitParameter("Site"); 4 // 输出初始化参数 5 System.out.println("网站名称: " + site); 6 }
web.xml配置各节点说明
<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。用户可以设置多个子元素用来指定 Filter 对资源的多种调用方式进行拦截。
Servlet 异常处理
当一个 Servlet 抛出一个异常时,Web 容器在使用了 exception-type 元素的 web.xml 中搜索与抛出异常类型相匹配的配置。web.xml 中使用 error-page 元素来指定对特定异常 或 HTTP 状态码 作出相应的 Servlet 调用。
web.xml 配置
假设,有一个 ErrorHandler 的 Servlet 在任何已定义的异常或错误出现时被调用。以下将是在 web.xml 中创建的项。
1 <!-- servlet 定义 --> 2 <servlet> 3 <servlet-name>ErrorHandler</servlet-name> 4 <servlet-class>ErrorHandler</servlet-class> 5 </servlet> 6 <!-- servlet 映射 --> 7 <servlet-mapping> 8 <servlet-name>ErrorHandler</servlet-name> 9 <url-pattern>/ErrorHandler</url-pattern> 10 </servlet-mapping> 11 12 <!-- error-code 相关的错误页面 --> 13 <error-page> 14 <error-code>404</error-code> 15 <location>/ErrorHandler</location> 16 </error-page> 17 <error-page> 18 <error-code>403</error-code> 19 <location>/ErrorHandler</location> 20 </error-page> 21 22 <!-- exception-type 相关的错误页面 --> 23 <error-page> 24 <exception-type>javax.servlet.ServletException</exception-type > 25 <location>/ErrorHandler</location> 26 </error-page> 27 <error-page> 28 <exception-type>java.io.IOException</exception-type > 29 <location>/ErrorHandler</location> 30 </error-page>
如果您想对所有的异常有一个通用的错误处理程序,那么应该定义下面的 error-page,而不是为每个异常定义单独的 error-page 元素:
1 <error-page> 2 <exception-type>java.lang.Throwable</exception-type > 3 <location>/ErrorHandler</location> 4 </error-page>