Filter

1、功能

  • Java Servlet 2.3 中新增加的功能,主要作用是对Servlet 容器的请求和响应进行检查和修改
  • Filter 本身并不生成请求和响应对象,它只提供过滤作用

    在Servlet 被调用之前,检查Request 对象
      »可以对其Request Header 和Request 内容进行审查和修改
    在Servlet 调用结束之后,检查Response 对象
      »可以对其Response Header 和Response 内容进行审查和修改

  • Filter 可以过滤的Web 组件包括Servlet,JSP,HTML等
  • Filter主要负责拦截请求,和放行。
  • Filter 过滤过程

2、接口Filter

  • init( FilterConfig config )

    Filter 的初始化方法
    容器创建Filter 之后将调用这个方法
    使用这个方法可以读取web.xml 文件中定义的初始化参数

  • doFilter(ServletRequest req,ServletResponse resp ,FilterChain chain)

    该方法完成实际的过滤操作
    当客户请求访问与Filter 相关联的URL 时,将调用该方法
    chain 用于访问后续的Filter 或Servlet

  • destroy()

    容器在销毁Filter 实例前调用该方法
    该方法中可以释放该Filter 所占用的资源

  • Filter的生命周期

    Filter的创建和销毁由web服务器控制。

      »服务器启动的时候,web服务器创建Filter的实例对象,并调用其init方法,完成对象的初始化功能。filter对象只会创建一次,init方法也只会执行一次。

      »拦截到请求时,执行doFilter方法。可以执行多次。

      »服务器关闭时,web服务器销毁Filter的实例对象。

3、接口FilterChain

  • 过滤器链
  • 一组过滤器对某些web资源进行拦截,那么这组过滤器就称为过滤器链。过滤器的执行顺序和<filter-mapping>有关

  • 该接口中定义的方法

    doFilter(ServletRequest req,ServletResponse resp )
      »负责把所有的过滤器给串联起来
      »使得一个过滤器执行完后,下一个可以继续执行
      »被串联的多个过滤器按照配置文件中的映射顺序依次执行

4、创建Filter

  • 实现Filter
package ecut.filter.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; public class HelloFilter implements Filter { public void init(FilterConfig config) throws ServletException {
} public void doFilter(ServletRequest req, ServletResponse resp, FilterChain chain) throws IOException, ServletException {
// place your code here
System.out.println( "hello" ); // pass the request along the filter chain
chain.doFilter( req, resp ); System.out.println( "world" );
} public void destroy() {
} }
  • 注册Filter
<filter>
<filter-name>HelloFilter</filter-name>
<filter-class>ecut.filter.filter.HelloFilter</filter-class>
</filter>
  • 发布Filter

    同Servlet 一样,url-pattern 可以写多个

    url-mapping匹配规则有三种:

      »精确匹配 —— 如/index.html,只会匹配index.html这个URL

      »路径匹配 —— 如/*,会匹配所有的URL

      »后缀匹配 —— 如*.html,会匹配所有以.html为后缀的URL

<filter-mapping>
<filter-name>HelloFilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
  • 测试案例
package ecut.filter.servlet;

import java.io.IOException;
import java.io.PrintWriter; import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse; public class HelloServlet extends HttpServlet { private static final long serialVersionUID = -8731391727918781480L; @Override
protected void service( HttpServletRequest request, HttpServletResponse response )
throws ServletException, IOException { System.out.println( "service" ); response.setContentType( "text/html" ); PrintWriter w = response.getWriter(); w.println( "<h2 style='text-align:center;'>你好 , Servlet .</h2>" ); } }
<servlet>
<servlet-name>HelloServlet</servlet-name>
<servlet-class>ecut.filter.servlet.HelloServlet</servlet-class>
</servlet> <servlet-mapping>
<servlet-name>HelloServlet</servlet-name>
<url-pattern>/filter/hello</url-pattern>
</servlet-mapping>

在浏览器中访问HelloServle

运行结果如下:

hello
service
world

5、在config中指定初始化参数

  • 用 户在配置filter时,可以使用<init-param>为filter配置一些初始化参数,当web容器实例化Filter对象,调用其 init方法时,会把封装了filter初始化参数的filterConfig对象传递进来
  • 在web.xml 中可以指定初始化参数
<filter>
<filter-name>CharacterEncodingFilter</filter-name>
<filter-class>ecut.filter.filter.CharacterEncodingFilter</filter-class>
<init-param>
<param-name>encoding</param-name>
<param-value>UTF-8</param-value>
</init-param>
</filter>
  • 访问初始化参数

    在init 方法中通过FilterConfig 对象访问

字符集过滤器测试案例:

package ecut.filter.filter;

import java.io.IOException;
import java.nio.charset.Charset; import javax.servlet.DispatcherType;
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; public class CharacterEncodingFilter implements Filter{ private static final String ENCODING_PARAM = "encoding" ; private static final String DEFAULT_ENCODING = "UTF-8" ; private String encoding ; @Override
public void init( FilterConfig filterConfig ) throws ServletException {
System.out.println( "CharacterEncodingFilter 初始化" );
// 获取 Filter 的初始化参数
encoding = filterConfig.getInitParameter( ENCODING_PARAM );
// 如果 未指定初始化参数 或 初始化参数值是 空串 则采用默认编码
encoding = ( encoding == null || encoding.trim().isEmpty() ) ? DEFAULT_ENCODING : encoding ;
// 如果指定的编码名称不被JVM所支持,则采用默认编码
// encoding = Charset.isSupported( encoding ) ? encoding : DEFAULT_ENCODING ;
/*
if( ! Charset.isSupported( encoding ) ) { // 如果指定的字符集名称是不支持的
encoding = DEFAULT_ENCODING ;
}
*/
} @Override
public void doFilter( ServletRequest req , ServletResponse resp , FilterChain chain )
throws IOException, ServletException {
HttpServletRequest request = (HttpServletRequest) req ;
String uri = request.getRequestURI();
DispatcherType type = request.getDispatcherType();
System.out.println( "CharacterEncodingFilter ::: DispatcherType : " + type + " , URI : " + uri );
// place your code here
req.setCharacterEncoding( encoding );
resp.setCharacterEncoding( encoding );
// pass the request along the filter chain
chain.doFilter( req , resp );
} @Override
public void destroy() {
System.out.println( "CharacterEncodingFilter 销毁" );
} }

运行结果:

解决中文乱码问题,使用过滤器后还需要 response.setContentType( "text/html" );否则仍然会乱码, 或者在响应中文中直接指定字符集response.setContentType( "text/html;charset=UTF-8" );
6、实现对指定的调度进行过滤

  • 在web.xml 中可以指定过滤的请求REQUEST:直接访问目标资源时执行过滤器。包括:在地址栏中直接访问、表单提交、超链接、重定向,只要在地址栏中可以看到目标资源的路径,就是REQUEST;
  • FORWARD:转发访问执行过滤器。包括RequestDispatcher#forward()方法、< jsp:forward>标签都是转发访问;
  • INCLUDE:包含访问执行过滤器。包括RequestDispatcher#include()方法、< jsp:include>标签都是包含访问;
  • ERROR:当目标资源在web.xml中配置为< error-page>中时,并且真的出现了异常,转发到目标资源时,会执行过滤器。
<filter-mapping>
<filter-name>CharacterEncodingFilter</filter-name>
<url-pattern>/*</url-pattern>
<dispatcher>REQUEST</dispatcher>
<dispatcher>FORWARD</dispatcher>
<dispatcher>INCLUDE</dispatcher>
</filter-mapping>
  • 使得filter将会作用于直接从客户端过来的request,通过forward过来的request,通过include过来的request和通过<error-page>过来的request。如果没有指定任何< dispatcher >元素,默认值是REQUEST。

测试案例:

package ecut.filter.servlet;

import java.io.IOException;

import javax.servlet.RequestDispatcher;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse; @WebServlet( "/filter/action/login" )
public class LoginActionServlet extends HttpServlet { private static final long serialVersionUID = -7857947923197325636L; @Override
protected void service( HttpServletRequest request, HttpServletResponse response )
throws ServletException, IOException { String username = request.getParameter( "username" );
String password = request.getParameter( "password" ); System.out.println( "username : " + username + " , password : " + password ); request.setAttribute( "username" , username ); RequestDispatcher dispatcher = request.getRequestDispatcher( "/filter/success/login" ); dispatcher.forward( request, response ); } }
package ecut.filter.servlet;

import java.io.IOException;

import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse; @WebServlet( "/filter/success/login" )
public class LoginSuccessServlet extends HttpServlet { private static final long serialVersionUID = 2707448998600792264L; @Override
protected void service( HttpServletRequest request, HttpServletResponse response )
throws ServletException, IOException { //String username = (String)request.getAttribute( "username" ); String username = request.getParameter( "username" );
System.out.println( "success : " + username ); response.setContentType( "text/html" ); response.getWriter().println( "<h1>" + username + " </h1>" ); } }
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>Filter</title>
</head>
<body> <h5>登录</h5>
<form action="/s/action/login" method="post" >
<input type="text" name="username" placeholder="用户名">
<input type="password" name="password" placeholder="输入密码">
<input type="submit" value="登录">
</form> </body>
</html>

运行结果:

hello
CharacterEncodingFilter ::: DispatcherType : REQUEST , URI : /s/filter/action/login
IdentifyBrowserFilter ::: User Agent : mozilla/5.0 (windows nt 6.1; win64; x64; rv:59.0) gecko/20100101 firefox/59.0
username : 郑 , password : 123
CharacterEncodingFilter ::: DispatcherType : FORWARD , URI : /s/filter/success/login
success : 郑
world

8、使用注解

  • Servlet 3.0 允许使用注解来标注Filter
  • @WebFilter用以标注一个实现过Filter 接口的类
  • 使用@WebFilter 标注不如web.xml 文件中可以通过映射顺序来控制过滤器的执行顺序
  • @WebFilter 的常用属性

    String filterName 指定当前Filter 的名称,相当于xml 中的filter-name
    String[] value 指定当前Filter 对应的url (与url-pattern 对应)
    String[] urlPatterns 与value 作用相同
    DispatcherType[] dispatcherTypes 指定当前Filter 关联的dispatcher 类型
      »默认值是DispatcherType.REQUEST
      »javax.servlet.DispatcherType 是Servlet 3.0 新定义的枚举
    boolean asyncSupported 指定是否支持异步操作
    WebInitParam[] initParams 用于设置Filter 初始化参数

9、使用过滤器过滤浏览器

测试案例:

package ecut.filter.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.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse; public class IdentifyBrowserFilter implements Filter { @Override
public void init( FilterConfig config ) throws ServletException {
} @Override
public void doFilter( ServletRequest req , ServletResponse resp , FilterChain chain )
throws IOException, ServletException { HttpServletRequest request = (HttpServletRequest) req ;
HttpServletResponse response = (HttpServletResponse) resp ; String userAgent = request.getHeader( "user-agent" ); userAgent = userAgent.toLowerCase(); System.out.println( "IdentifyBrowserFilter ::: User Agent : " + userAgent ); // 如果在 userAgent 中找到了 trident 则说明目前正在使用 IE 访问
if( userAgent.indexOf( "trident" ) != - 1 ) {
String uri = request.getRequestURI();
System.out.println( "uri : " + uri );
int index = uri.lastIndexOf( "/" );
uri = uri.substring( 0 , index ) ;
System.out.println( "uri : " + uri ); index = uri.lastIndexOf( "/" );
uri = uri.substring( index ) ;
System.out.println( "uri : " + uri );
//解析字符串,如果访问的是IE目录下的就继续
if( "/ie".equals( uri ) ) {
chain.doFilter( req , resp );
} else {//如果访问的不是IE目录下的就重定向到ie.html
response.sendRedirect( request.getContextPath() + "/pages/filter/ie/ie.html" );
} } else {
// 如果是 非 IE 浏览器,则可以通过 FilterChain 向后传递 请求 和 响应
chain.doFilter( req , resp );
} } @Override
public void destroy() {
} }
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>珍爱生命,远离IE</title>
<link rel="stylesheet" href="/s/pages/filter/ie/ie.css" >
</head>
<body> <h2>珍爱生命,远离IE</h2> <h2>请使用现代浏览器: Chrome 、FireFox </h2> </body>
</html>
@CHARSET "UTF-8";

h2 {
text-align: center;
} h2:first-child {
color: red ;
font-weight: bold ;
} h2:last-child {
color: blue ;
}

运行结果:

转载请于明显处标明出处

http://www.cnblogs.com/AmyZheng/p/9024091.html

Servlet 学习(八)的更多相关文章

  1. Servlet学习(八)——Session

    1.Session技术 Session技术是将数据存储在服务器端的技术,会为每个客户端都创建一块内存空间存储客户的数据,但客户端需要每次都携带一个标识ID去服务器中寻找属于自己的内存空间.所以说Ses ...

  2. JSP&Servlet学习手册

    JSP&Servlet学习手册 沙琪玛 书 目录 JSP 指令... 3 书写方式... 3 指令列表... 3 JSP 内置对象... 3 内置对象特点... 3 常用内置对象... 3 o ...

  3. Python Tutorial 学习(八)--Errors and Exceptions

    Python Tutorial 学习(八)--Errors and Exceptions恢复 Errors and Exceptions 错误与异常 此前,我们还没有开始着眼于错误信息.不过如果你是一 ...

  4. Servlet 学习笔记

    Servlet 运行在服务器上的 java 类: Servlet 容器为 javaWeb 应用提供运行时环境,负责管理 servlet 和 jsp 生命周期,以及管理他们的共享数据. 现在我们知道了 ...

  5. Servlet学习:(三)Servlet3.0 上传文件

    转: Servlet学习:(三)Servlet3.0 上传文件 2018年08月03日 11:57:58 iDark_CSDN 阅读数:362   一.注意事项 客户端(浏览器) 表单的提交方法必须是 ...

  6. SVG 学习<八> SVG的路径——path(2)贝塞尔曲线命令、光滑贝塞尔曲线命令

    目录 SVG 学习<一>基础图形及线段 SVG 学习<二>进阶 SVG世界,视野,视窗 stroke属性 svg分组 SVG 学习<三>渐变 SVG 学习<四 ...

  7. Servlet学习(九)——request

    request运行流程在Servlet学习(四)——response已介绍,不再赘述 1.通过抓包工具获取Http请求 因为request代表请求,所以我们可以通过该对象分别获得Http请求的请求行, ...

  8. # jsp及servlet学习笔记

    目录 jsp及servlet学习笔记 JSP(Java Server Page Java服务端网页) 指令和动作: servlet(小服务程序) jsp及servlet学习笔记 JSP(Java Se ...

  9. Servlet学习笔记(四)

    目录 Servlet学习笔记(四) 一.会话技术Cookie.session 1. 什么是会话技术? 2. 会话技术有什么用? 3. Cookie 3.1 什么是Cookie? 3.2 使用Cooki ...

  10. Servlet学习笔记(三)

    目录 Servlet学习笔记(三) 一.HTTP协议 1.请求:客户端发送欸服务器端的数据 2.响应:服务器端发送给客户端的数据 3.响应状态码 二.Response对象 1.Response设置响应 ...

随机推荐

  1. Git - 07. gitignore

    1. 概述 开发的过程中, 无法保证项目文件夹下的所有东西, 都是想传到版本库的 比如 maven 项目的 target 目录 配置好之后, 使用 git add .命令, 这些文件\目录也不会被 s ...

  2. 433B.Kuriyama Mirai's Stones

    Kuriyama Mirai has killed many monsters and got many (namely n) stones. She numbers the stones from  ...

  3. STA之AOCV

    为什么要引入AOCV 为了精确性,为了剔除悲观度.用set_timing_derate来设置OCV,对于一个固定的corner,只能对data/clock, cell/net, late/early分 ...

  4. JEECG屏蔽在线聊天插件

    如图所示: 找到pom.xml文件将如下代码屏蔽即可: <!-- 在线聊天工具 --> <dependency> <groupId>org.p3framework& ...

  5. Dart语言学习(五)Dart Bool类型

    Dart Bool类型和其他语言类似,比较简单 其特点有: 1.使用 bool 表示布尔类型 2.布尔值只有 true 和 false 3.布尔类型bool默认值是null bool isTrue = ...

  6. 前端开发CSS清除浮动的方法有哪些?

    在前端开发过程中,非IE浏览器下,当容器的高度自动,并且容器内容中有浮动元素(float为left或right),此时如果容器的高度不能自适应内容的高度,从而使得内容溢出破坏整体布局,这种现象叫做浮动 ...

  7. 一些封装的php函数

    swoole群中奥总共享的创建文件夹: function make_dir($folder){ $reval = false; if (!file_exists($folder)){ /* 如果目录不 ...

  8. Go语言基础之runtime包

    文章引用自 Golang中runtime的使用 runtime调度器是非常有用的东西,关于runtime包几个方法: Gosched:让当前线程让出cpu以让其他线程运行,它不会挂起当前线程,因此当前 ...

  9. Vue项目中sass语法该怎么用?

    最近开始着手Vue框架,被各种报错蹂躏,其中有一个就是sass语法,<style>标签中添加<style lang="scss">,发现报错,在网上找了一些 ...

  10. JavaScript的技巧45招

    JavaScript奇技淫巧45招 来自仲老师的分享: 原文地址[http://chensd.com/2015-01/45-useful-javascript-tips-tricks-and-best ...