一、Servlet过滤器
用来拦截请求和响应,使用其中信息,在客户端请求后端资源之前,拦截请求,在服务端将响应发送给客户端之前,处理响应。
怎样实现过滤器呢
首先新建一个类实现javax.servlet.Filter接口,该类中有三个方法
init() 初始化操作,匹配web.xml中的过滤标签,并获取初始化数据,只执行一次
doFilter() 匹配成功时,执行具体操作。请求一次,执行一次。
destory() 在Servlet容器销毁过滤实例之前调用,释放资源。
接着要在web.xml中建立过滤器的映射
在<servlet>之前使用<filter>和<filter-mapping>
<filter>
<filter-name>filter</filter-name> <!-- 过滤器名 -->
<filter-class>com.demo.FilterFirst</filter-class> <!-- 过滤器的类名 -->
<init-param>
<param-name>name</param-name> <!-- 初始化参数名-->
<param-value>filter</param-value> <!-- 初始化参数值-->
</init-param>
</filter>
<filter-mapping>
<filter-name>filter</filter-name> <!-- 过滤器名 -->
<url-pattern>/*</url-pattern> <!-- 过滤器所适用的Servlet -->
</filter-mapping>
其中
<url-pattern>/*</url-pattern> /*是指适用于所有的Servlet,如果要过滤指定的servlet ,可以修改到其具体路径
具体示例
package com.demo;
import java.io.IOException;
import javax.servlet.*;
public class FilterFirst implements Filter{
@Override
public void init(FilterConfig config) throws ServletException {
// TODO Auto-generated method stub
// 初始化数据
String name=config.getInitParameter("name");
System.out.println("获取 "+name);
}
@Override
public void doFilter(ServletRequest request, ServletResponse reponse, FilterChain chain)
throws IOException, ServletException {
// TODO Auto-generated method stub
// 满足条件,请求一次执行一次
System.out.println("name filter");
// 把请求传回过滤链
chain.doFilter(request, reponse);
}
@Override
public void destroy() {
// TODO Auto-generated method stub
}
}
<?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">
<filter>
<filter-name>filter</filter-name> <!-- 过滤器名 -->
<filter-class>com.demo.FilterFirst</filter-class> <!-- 过滤器的类名 -->
<init-param>
<param-name>name</param-name> <!-- 初始化参数名-->
<param-value>filter</param-value> <!-- 初始化参数值-->
</init-param>
</filter>
<filter-mapping>
<filter-name>filter</filter-name> <!-- 过滤器名 -->
<url-pattern>/*</url-pattern> <!-- 过滤器所适用的Servlet -->
</filter-mapping>
<servlet>
<servlet-name>test</servlet-name> <!-- 自定义name -->
<servlet-class>com.demo.MServlet</servlet-class> <!-- 完整包名+类名 -->
</servlet>
<servlet-mapping>
<servlet-name>test</servlet-name> <!-- 同上 -->
<url-pattern>/MServlet</url-pattern> <!-- 访问的url,去掉了上下文环境 -->
</servlet-mapping>
<servlet-mapping>
<servlet-name>test</servlet-name> <!-- 同上 -->
<url-pattern>/other/MServlet</url-pattern> <!-- 访问的url,去掉了上下文环境 -->
</servlet-mapping>
</web-app>
在浏览器中输入具Servlet的url,Console中看到
请求一次,doFilter方法执行一次。
当然一个web.xml中可以有多个过滤器,执行顺序有<filter-mapping>的顺序来决定
二、Servlet异常处理
Servlet抛出异常时,web容器通过error-code和exception-type在web.xml中配置的错误和异常信息匹配,匹配后跳转到制定页面
首先在web.xml中配置
<error-page>
<error-code>404</error-code> <!-- 错误 -->
<location>/MServlet</location> <!-- 错误时调用 -->
</error-page>
<error-page>
<exception-type>java.lang.Throwable</exception-type> <!-- 异常类型 -->
<location>/MServlet</location> <!--异常时调用 -->
</error-page>
java.lang.Throwable 是一个通用的异常
以下为Servlet中用来处理错误和异常的属性
在doGet方法中处理异常。
// 处理错误和异常
Throwable throwable = (Throwable)request.getAttribute("javax.servlet.error.exception");
Integer statusCode = (Integer)request.getAttribute("javax.servlet.error.status_code");
String servletName = (String)request.getAttribute("javax.servlet.error.servlet_name");
if (servletName == null){
servletName = "Unknown";
}
String requestUri = (String)request.getAttribute("javax.servlet.error.request_uri");
if (requestUri == null){
requestUri = "Unknown";
}
if (throwable == null && statusCode == null){
writer.println("错误信息丢失");
writer.println("请返回"+
response.encodeURL("http://localhost:8080/"));
}else if (statusCode != null) {
writer.println("错误代码 : " + statusCode);
}else{
writer.println("错误信息");
writer.println("Servlet Name : " + servletName);
writer.println("异常类型 : " +throwable.getClass( ).getName( ));
writer.println("请求 URI: " + requestUri);
writer.println("异常信息: " + throwable.getMessage( ));
}