首先来看一下Servlet的过滤器内容:

一、Servlet过滤器的概念:

***************************************************************************************

Servlet过滤器是在Java Servlet规范2.3中定义的,它能够对Servlet容器的请求和响应对象进行检查和修改。   

Servlet过滤器本身并不产生请求和响应对象,它只能提供过滤作用。Servlet过期能够在Servlet被调用之前检查Request对象,修改Request Header和Request内容;在Servlet被调用之后检查Response对象,修改Response Header和Response内容。

Servlet过期负责过滤的Web组件可以是Servlet、JSP或者HTML文件。 

***************************************************************************************



二、Servlet过滤器的特点:

***************************************************************************************

A.Servlet过滤器可以检查和修改ServletRequest和ServletResponse对象

B.Servlet过滤器可以被指定和特定的URL关联,只有当客户请求访问该URL时,才会触发过滤器

C.Servlet过滤器可以被串联在一起,形成管道效应,协同修改请求和响应对象

***************************************************************************************



三、Servlet过滤器的作用:

***************************************************************************************

A.查询请求并作出相应的行动。

B.阻塞请求-响应对,使其不能进一步传递。

C.修改请求的头部和数据。用户可以提供自定义的请求。

D.修改响应的头部和数据。用户可以通过提供定制的响应版本实现。

E.与外部资源进行交互。

***************************************************************************************



四、Servlet过滤器的适用场合:

***************************************************************************************

A.认证过滤

B.登录和审核过滤

C.图像转换过滤 

D.数据压缩过滤 

E.加密过滤 

F.令牌过滤 

G.资源访问触发事件过滤 

H.XSL/T过滤 

I.Mime-type过滤

***************************************************************************************



五、Servlet过滤器接口的构成:

***************************************************************************************

所有的Servlet过滤器类都必须实现javax.servlet.Filter接口。这个接口含有3个过滤器类必须实现的方法:

A.init(FilterConfig):

这是Servlet过滤器的初始化方法,Servlet容器创建Servlet过滤器实例后将调用这个方法。在这个方法中可以读取web.xml文件中Servlet过滤器的初始化参数

B.doFilter(ServletRequest,ServletResponse,FilterChain):

这个方法完成实际的过滤操作,当客户请求访问于过滤器关联的URL时,Servlet容器将先调用过滤器的doFilter方法。FilterChain参数用于访问后续过滤器

C.destroy():

Servlet容器在销毁过滤器实例前调用该方法,这个方法中可以释放Servlet过滤器占用的资源

***************************************************************************************



六、Servlet过滤器的创建步骤:

***************************************************************************************

A.实现javax.servlet.Filter接口

B.实现init方法,读取过滤器的初始化函数

C.实现doFilter方法,完成对请求或过滤的响应

D.调用FilterChain接口对象的doFilter方法,向后续的过滤器传递请求或响应

E.销毁过滤器

***************************************************************************************



七、Servlet过滤器对请求的过滤:

***************************************************************************************

A.Servlet容器创建一个过滤器实例

B.过滤器实例调用init方法,读取过滤器的初始化参数

C.过滤器实例调用doFilter方法,根据初始化参数的值判断该请求是否合法

D.如果该请求不合法则阻塞该请求

E.如果该请求合法则调用chain.doFilter方法将该请求向后续传递

***************************************************************************************



八、Servlet过滤器对响应的过滤:

***************************************************************************************

A.过滤器截获客户端的请求

B.重新封装ServletResponse,在封装后的ServletResponse中提供用户自定义的输出流

C.将请求向后续传递

D.Web组件产生响应

E.从封装后的ServletResponse中获取用户自定义的输出流

F.将响应内容通过用户自定义的输出流写入到缓冲流中

G.在缓冲流中修改响应的内容后清空缓冲流,输出响应内容

***************************************************************************************



九、Servlet过滤器的发布:

***************************************************************************************

A.发布Servlet过滤器时,必须在web.xml文件中加入<filter>元素和<filter-mapping>元素。

B.<filter>元素用来定义一个过滤器:

属性                   含义

filter-name    指定过滤器的名字

filter-class    指定过滤器的类名

init-param    为过滤器实例提供初始化参数,可以有多个

C.<filter-mapping>元素用于将过滤器和URL关联:

属性                     含义

filter-name    指定过滤器的名字

url-pattern    指定和过滤器关联的URL,为”/*”表示所有URL

***************************************************************************************



十一、Servlet过滤器使用的注意事项

***************************************************************************************

A.由于Filter、FilterConfig、FilterChain都是位于javax.servlet包下,并非HTTP包所特有的,所以其中所用到的请求、响应对象ServletRequest、ServletResponse在使用前都必须先转换成HttpServletRequest、HttpServletResponse再进行下一步操作。

B.在web.xml中配置Servlet和Servlet过滤器,应该先声明过滤器元素,再声明Servlet元素

C.如果要在Servlet中观察过滤器生成的日志,应该确保在server.xml的localhost对应的<host>元素中配置如下<logger>元素:

<Logger className = “org.apache.catalina.logger.FileLogger”

directory = “logs”prefix = “localhost_log.”suffix=”.txt”

timestamp = “true”/>

***************************************************************************************

十二、一个实例

首先来看一下web.xml的配置:

<!-- 请求url日志记录过滤器 -->
<filter>
<filter-name>logfilter</filter-name>
<filter-class>com.weijia.filterservlet.LogFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>logfilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping> <!-- 编码过滤器 -->
<filter>
<filter-name>setCharacterEncoding</filter-name>
<filter-class>com.weijia.filterservlet.EncodingFilter</filter-class>
<init-param>
<param-name>encoding</param-name>
<param-value>utf-8</param-value>
</init-param>
</filter>
<filter-mapping>
<filter-name>setCharacterEncoding</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>

然后看一下编码过滤器:

package com.weijia.filterservlet;

import java.io.IOException;
import java.util.Enumeration;
import java.util.HashMap; 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 EncodingFilter implements Filter {
private String encoding;
private HashMap<String,String> params = new HashMap<String,String>();
// 项目结束时就已经进行销毁
public void destroy() {
System.out.println("end do the encoding filter!");
params=null;
encoding=null;
}
public void doFilter(ServletRequest req, ServletResponse resp,FilterChain chain) throws IOException, ServletException {
System.out.println("before encoding " + encoding + " filter!");
req.setCharacterEncoding(encoding);
chain.doFilter(req, resp);
System.out.println("after encoding " + encoding + " filter!");
System.err.println("----------------------------------------");
} // 项目启动时就已经进行读取
public void init(FilterConfig config) throws ServletException {
System.out.println("begin do the encoding filter!");
encoding = config.getInitParameter("encoding");
for (Enumeration<?> e = config.getInitParameterNames(); e.hasMoreElements();) {
String name = (String) e.nextElement();
String value = config.getInitParameter(name);
params.put(name, value);
}
}
}

日志过滤器:

package com.weijia.filterservlet;

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; public class LogFilter implements Filter { public FilterConfig config; public void destroy() {
this.config = null;
System.out.println("end do the logging filter!");
} public void doFilter(ServletRequest req, ServletResponse res, FilterChain chain) throws IOException, ServletException {
System.out.println("before the log filter!");
// 将请求转换成HttpServletRequest 请求
HttpServletRequest hreq = (HttpServletRequest) req;
// 记录日志
System.out.println("Log Filter已经截获到用户的请求的地址:"+hreq.getServletPath() );
try {
// Filter 只是链式处理,请求依然转发到目的地址。
chain.doFilter(req, res);
} catch (Exception e) {
e.printStackTrace();
}
System.out.println("after the log filter!");
} public void init(FilterConfig config) throws ServletException {
System.out.println("begin do the log filter!");
this.config = config;
} }

测试Servlet:

package com.weijia.filterservlet;

import java.io.IOException;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse; public class FilterServlet extends HttpServlet { private static final long serialVersionUID = 1L; public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
response.setDateHeader("expires", -1);
} public void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
} }

访问FilterServlet

运行结果:

before the log filter!

Log Filter已经截获到用户的请求的地址:/FilterServlet

before encoding utf-8 filter!

after encoding utf-8 filter!

----------------------------------------

after the log filter!

我们从运行结果可以看到这个过滤器的调用关系:

类似于C++中的构造函数和析构函数的调用顺序,

这里我们在web.xml中注册的是先注册日志过滤器的,然后再注册

当我们重新部署应用的时候发现:

会先销毁上次的过滤器,然后再重新注册一下

下面在来看一下Servlet的监听器

Servlet监听器用于监听一些重要事件的发生,监听器对象可以在事情发生前、发生后可以做一些必要的处理。下面将介绍几种常用的监听器,以及它们都适合运用于那些环境。

分类及介绍:

1.  ServletContextListener:用于监听WEB 应用启动和销毁的事件,监听器类需要实现javax.servlet.ServletContextListener 接口。

  1. public class QuartzListener implements ServletContextListener {
  2. private Logger logger = LoggerFactory.getLogger(QuartzListener.class);
  3. public void contextInitialized(ServletContextEvent sce) {
  4. }
  5. /**
  6. *在服务器停止运行的时候停止所有的定时任务
  7. */
  8. @SuppressWarnings("unchecked")
  9. public void contextDestroyed(ServletContextEvent arg0) {
  10. try {
  11. Scheduler scheduler = StdSchedulerFactory.getDefaultScheduler();
  12. List<JobExecutionContext> jobList = scheduler.getCurrentlyExecutingJobs();
  13. for (JobExecutionContext jobContext : jobList) {
  14. Job job = jobContext.getJobInstance();
  15. if (job instanceof InterruptableJob) {
  16. ((InterruptableJob) job).interrupt();
  17. }
  18. }
  19. scheduler.shutdown();
  20. } catch (SchedulerException e) {
  21. logger.error("shut down scheduler happened error", e);
  22. }
  23. }
  24. }

2.  ServletContextAttributeListener:用于监听WEB应用属性改变的事件,包括:增加属性、删除属性、修改属性,监听器类需要实现javax.servlet.ServletContextAttributeListener接口。 



3.  HttpSessionListener:用于监听Session对象的创建和销毁,监听器类需要实现javax.servlet.http.HttpSessionListener接口或者javax.servlet.http.HttpSessionActivationListener接口,或者两个都实现。

  1. /**
  2. *
  3. * 会话监听器
  4. * <p />
  5. *
  6. */
  7. public class SessionListener implements HttpSessionListener {
  8. @Override
  9. public void sessionCreated(HttpSessionEvent arg0) {
  10. }
  11. @Override
  12. public void sessionDestroyed(HttpSessionEvent event) {
  13. HttpSession session = event.getSession();
  14. User user = (BrsSession) session.getAttribute("currUser");
  15. if (user != null) {
  16. //TODO something
  17. }
  18. }
  19. }

4.  HttpSessionActivationListener:用于监听Session对象的钝化/活化事件,监听器类需要实现javax.servlet.http.HttpSessionListener接口或者javax.servlet.http.HttpSessionActivationListener接口,或者两个都实现。 



5.  HttpSessionAttributeListener:用于监听Session对象属性的改变事件,监听器类需要实现javax.servlet.http.HttpSessionAttributeListener接口。 



部署: 

       监听器的部署在web.xml文件中配置,在配置文件中,它的位置应该在过滤器的后面Servlet的前面

web.xml配置文件:

  1. <!-- Quartz监听器 -->
  2. <listener>
  3. <listener-class>
  4. com.flyer.lisenter.QuartzListener
  5. </listener-class>
  6. </listener>

JavaWeb学习篇之----Servlet过滤器Filter和监听器的更多相关文章

  1. Servlet过滤器Filter和监听器

    一.Servlet过滤器的概念: *********************************************************************************** ...

  2. JavaWeb学习篇之----Servlet

    今天来继续学习JavaWeb的相关知识,之前都是都介绍一些基本知识,从今天开始我们来说一下如何在服务器编写程序,这里就需要来介绍一下Servlet的相关知识了.Servlet就是一个能够运行在服务器端 ...

  3. JavaWeb学习篇之----自定义标签&&JSTL标签库详解

    今天来看一下自定义标签的内容,自定义标签是JavaWeb的一部分非常重要的核心功能,我们之前就说过,JSP规范说的很清楚,就是Jsp页面中禁止编写一行Java代码,就是最好不要有Java脚本片段,下面 ...

  4. JavaWeb学习篇--Filter过滤器

    Filter过滤器简介 ServletAPI中提供了一个Filter接口,开发web应用时,如果编写的 java 类实现了这个接口,则把这个java类称之为过滤器Filter. WEB服务器每次在调用 ...

  5. javaWeb学习总结(10)- Filter(过滤器)常见应用(3)

    一.统一全站字符编码 通过配置参数charset指明使用何种字符编码,以处理Html Form请求参数的中文问题 package me.gacl.web.filter; import java.io. ...

  6. javaweb学习总结(四十二)——Filter(过滤器)学习

    一.Filter简介 Filter也称之为过滤器,它是Servlet技术中最激动人心的技术,WEB开发人员通过Filter技术,对web服务器管理的所有web资源:例如Jsp, Servlet, 静态 ...

  7. javaWeb学习总结(10)- Filter(过滤器)学习(2)

    在filter中可以得到代表用户请求和响应的request.response对象,因此在编程中可以使用Decorator(装饰器)模式对request.response对象进行包装,再把包装对象传给目 ...

  8. javaWeb学习总结(10)- Filter(过滤器)学习

    一.Filter简介 Filter也称之为过滤器,它是Servlet技术中最激动人心的技术,WEB开发人员通过Filter技术,对web服务器管理的所有 web资源:例如Jsp, Servlet, 静 ...

  9. JavaWeb学习(二十三)———Filter(过滤器)

    一.Filter简介 Filter也称之为过滤器,它是Servlet技术中最激动人心的技术,WEB开发人员通过Filter技术,对web服务器管理的所有web资源:例如Jsp, Servlet, 静态 ...

随机推荐

  1. 四大开源协议比较:BSD、Apache、GPL、LGPL(转)

    转自: 四大开源协议比较:BSD.Apache.GPL.LGPL 本文参考文献:http://www.fsf.org/licensing/licenses/ 现今存在的开源协议很多,而经过Open S ...

  2. SQL复制数据表及表结构

    select * into 目标表名 from 源表名 insert into 目标表名(fld1, fld2) select fld1, 5 from 源表名 以上两句都是将'源表'的数据插入到'目 ...

  3. Codeforces F. Bits And Pieces(位运算)

    传送门. 位运算的比较基本的题. 考虑枚举\(i\),然后二进制位从大到小考虑, 对于第\(w\)位,如果\(a[i][w]=1\),那么对\(j.k\)并没有什么限制. 如果\(a[i][w]=0\ ...

  4. python培训拾遗

    20140421 1. 三大利器: dir----列出所有内部方法 a=1 dir(a) 可以列出所有内部方法,就是带两个下划线的:带一个下划线的是普通方法 help---查看帮助 help(a.bi ...

  5. (转)OpenFire源码学习之二十七:Smack源码解析

    转:http://blog.csdn.net/huwenfeng_2011/article/details/43484199 Smack Smack是一个用于和XMPP服务器通信的类库,由此可以实现即 ...

  6. AutoCAD2016简体中文破解版32位64位下载

    AutoCAD2016序列号:666-69696969 667-98989898 400-45454545 066-66666666(任意一个) AutoCAD2016产品密钥:001H1 AutoC ...

  7. 二:unittest框架配合selenium之xpath定位

    刚开始学习selenium自动化测试时,犯了一个不该犯的错误,偷懒,使用火狐浏览器中的扩展FIREBUG,FIREPATH来辅助定位. 虽然用的定位方法大多数是使用XPATH方法,但是是工具定位出来的 ...

  8. (53)C# 工具

    https://docs.microsoft.com/zh-cn/dotnet/framework/tools/ildasm-exe-il-disassembler 一.Visual Studio的开 ...

  9. Java学习之集合(LinkedList链表集合)

    一.什么是链表集合,通过图形来看,比如33只知道它下一个是55 如果:现在要删除33的话,就是把55赋值给45,这样看它操作集合速度会非常快. 二.LinkedList特有方法 1.添加 addFir ...

  10. 1.6 USB的插入检测机制