java过滤器和监听器详解

news/2023/10/4 0:23:29

 过滤器

1Filter工作原理(执行流程)       

       当客户端发出Web资源的请求时,Web服务器根据应用程序配置文件设置的过滤规则进行检查,若客户请求满足过滤规则,则对客户请求/响应进行拦截,对请求头和请求数据进行检查或改动,并依次通过过滤器链,最后把请求/响应交给请求的Web资源处理。请求信息在过滤器链中可以被修改,也可以根据条件让请求不发往资源处理器,并直接向客户机发回一个响应。当资源处理器完成了对资源的处理后,响应信息将逐级逆向返回。同样在这个过程中,用户可以修改响应信息,从而完成一定的任务。

         上面说了,当一个请求符合某个过滤器的过滤条件时该请求就会交给这个过滤器去处理。那么当两个过滤器同时过滤一个请求时谁先谁后呢?这就涉及到了过滤链FilterChain

         所有的奥秘都在FilterFilterChain中。服务器会按照web.xml中过滤器定义的先后循序组装成一条链,然后一次执行其中的doFilter()方法。执行的顺序就如下图所示,执行第一个过滤器的chain.doFilter()之前的代码,第二个过滤器的chain.doFilter()之前的代码,请求的资源,第二个过滤器的chain.doFilter()之后的代码,第一个过滤器的chain.doFilter()之后的代码,最后返回响应。

        
        这里还有一点想补充:大家有没有想过,上面说的执行请求的资源究竟是怎么执行的?对于执行第一个过滤器的chain.doFilter()之前的代码,第二个过滤器的chain.doFilter()之前的代码这些我可以理解,无非就是按顺序执行一句句的代码,但对于这个执行请求的资源我刚开始却是怎么也想不明白。其实是这样的:

        通常我们所访问的资源是一个servletjsp页面,而jsp其实是一个被封装了的servlet,于是我们就可以统一地认为我们每次访问的都是一个Servlet,而每当我们访问一个servlet时,web容器都会调用该Servletservice方法去处理请求。而在service方法又会根据请求方式的不同(Get/Post)去调用相应的doGet()doPost()方法,实际处理请求的就是这个doGetdoPost方法。写过servlet的朋友都应该知道,我们在doGet(或doPost)方法中是通过response.getWriter()得到客户端的输出流对象,然后用此对象对客户进行响应。

       到这里我们就应该理解了过滤器的执行流程了:执行第一个过滤器的chain.doFilter()之前的代码——>第二个过滤器的chain.doFilter()之前的代码——>……——>n个过滤器的chain.doFilter()之前的代码——>所请求servletservice()方法中的代码——>所请求servletdoGet()doPost()方法中的代码——>n个过滤器的chain.doFilter()之后的代码——>……——>第二个过滤器的chain.doFilter()之后的代码——>第一个过滤器的chain.doFilter()之后的代码。

过滤器生命周期的四个阶段:

1、实例化:Web容器在部署Web应用程序时对所有过滤器进行实例化。Web容器回调它的无参构造方法。2、初始化:实例化完成之后,马上进行初始化工作。Web容器回调init()方法。

3、过滤:请求路径匹配过滤器的URL映射时。Web容器回调doFilter()方法——主要的工作方法。

4、销毁: Web容器在卸载Web应用程序前,Web容器回调destroy()方法。

Servlet过滤器开发步骤:

1、创建实现javax.servlet.Filter接口的类。

2、过滤器的xml配置。

Servlet过滤器API
 Servlet过滤器API包含了3个接口,它们都在javax.servlet包中,分别是Filter接口、FilterChain接口和FilterConfig接口。
public Interface Filter
所有的过滤器都必须实现Filter接口。该接口定义了init,doFilter0destory()三个方法:
  (1) public void init (FilterConfig filterConfig) 
当开始使用servlet过滤器服务时,Web容器调用此方法一次,为服务准备过滤器;然后在需要使用过滤器的时候调用doFilter(),传送给此方法的FilterConfig对象,包含servlet过滤器的初始化参数。
  (2)public void doFilter(ServletRequest requestServletResponse responseFilterChain chain)    
         每个过滤器都接受当前的请求和响应,且FilterChain过滤器链中的过滤器(应该都是符合条件的)都会被执行。doFilter方 法中,过滤器可以对请求和响应做它想做的一切,通过调用他们的方法收集数据,或者给对象添加新的行为。过滤器通过传送至 此方法的FilterChain参数,调用chaindoFilterO将控制权传送给下一个过滤器。当这个调用返回后,过滤器可以在它的 Filter方法的最后对响应做些其他的工作。如果过滤器想要终止请求的处理或得到对响应的完全控制,则可以不调用下一个过滤 器,而将其重定向至其它一些页面。当链中的最后一个过滤器调用chaindoFilterO方法时,将运行最初请求的Servlet
 (3)public void destroy()
       一旦doFilterO方法里的所有线程退出或已超时,容器调用
此方法。服务器调用destoryO以指出过滤器已结束服务,用于释
放过滤器占用的资源。
public interface FilterChain
public void doFilter(ServletRequest requestServletResponse response)
      此方法是由Servlet容器提供给开发者的,用于对资源请求过滤链的依次调用,通过FilterChain调用过滤链中的下一个过滤   器,如果是最后一个过滤器,则下一个就调用目标资源。
public interface FilterConfig
 FilterConfig接口检索过滤器名、初始化参数以及活动的Servlet上下文。该接口提供了以下4个方法:
     (1)public java1angString getFilterName0
           返回webxml部署文件中定义的该过滤器的名称。
     (2)public ServletContext getServletContextO
          返回调用者所处的servlet上下文。
     (3)public java.1ang.String getlnitParameter(java.1ang.String name)
返回过滤器初始化参数值的字符串形式,当参数不存在时,返回nul1name是初始化参数名。
     (4)public java.util.Enumeration getlnitParameterNames()
      Enumeration形式返回过滤器所有初始化参数值,如果没有初始化参数,返回为空。

三、应用实例
          从上面分析可知,实现Servlet过滤器,需要两步:第一步开发过滤器,设计个实现Fiker接口的类;第二步通过web.xml配置过滤器,实现过滤器和ServletJSP页面之间的映射。以下设计一个简单的IP地址过滤器,根据用户的IP地址进行对网站的访问控制。
(1)过滤器的设计ipfilter.java

[java] view plaincopy
print?
  1. package ipf;  
  2. imp0rt java.io.IOException;  
  3. imp0rt javax.servlet.*;  
  4. public class ipfilter implements Filter//实现Filter接口  
  5. {protected FilterConfig config;  
  6. protected String rejectedlP;  
  7. public void init(FilterConfig filterConfig)throws  
  8. ServletException  
  9.   
  10. {this.config=filterConfig;//从Web务器获取过滤器配置对象  
  11. rejectedlP=config.getlnitParameter( RejectedlP”):  
  12. //从配置中取得过滤lP  
  13. }  
  14. public void doFilter(ServletRequest request,  
  15. ServletResponse response.FilterChain chain)throws  
  16. IOException,ServletException  
  17. {RequestDispatcher dispatcher=request.getRequestDispatcher("");  
  18. String remotelP=request.getRemoteAddrO;//获取客户请求lP  
  19. int i=remotelP.1astlndexOf(".");  
  20. int r=rejectedlP.1astlndexOf(”.”):  
  21. String relPscope=rejectedlP.substring(0,r);//过滤lP段  
  22. if(relPscope.equals(remotelP.substring(O.i)))  
  23. {      dispatcher.forward(request,response);//重定向到rejectedError.jsp页面  
  24.         retum;//阻塞,直接返Web回客户端  
  25. }  
  26. else{chain.doFilter(request,response);//调用过滤链上的下一个过滤器  
  27. }  
  28. }  
  29. public void destroy()  



 

 //过滤器功能完成后,由Web服务器调用执行,回收过滤器资源
注意:chaindoFilterO语句以前的代码用于对客户请求的处理;以后的代码用于对响应进行处理。
(2)配置过滤器
    在应用程序Web—INF目录下的webxml描述符文件中添加以下代码:

[html] view plaincopy
print?
  1. <filter>  
  2. <filter-name>ipfIter</filter-name>//过滤器名称  
  3. <filter-class>ipf.ipfiIter</filter-class>//实现过滤器的类  
  4. <init—param>  
  5. <param—name>RejectedlP</param-name>//过滤器初始化参数名RejectedlP  
  6. <param-value>192.168.12.*/param-value>  
  7. </init—pamm>  
  8. </filter>  
  9. <filter-mapping>//过滤器映射(规律规则)  
  10. <filter-name>ipfiIter</filter-name>  
  11. <url—pattem>/*</ud-pattem>  
  12. //映射到Web应用根目录下的所有JSP文件  
  13. </filter-mapping>  


 

通过以上设计与配置,就禁止了IP地址处在192.168.12网段的用户对网站的访问。

监听器

一、监听器概述

监听你的web应用,监听许多信息的初始化,销毁,增加,修改,删除值等

Servlet监听器用于监听一些重要事件的发生,监听器对象可以在事情发生前、发生后可以做一些必要的处理。

  1.Listener是Servlet的监听器 

  2.可以监听客户端的请求、服务端的操作等。

  3.通过监听器,可以自动激发一些操作,如监听在线用户数量,当增加一个HttpSession时,给在线人数加1。

  4.编写监听器需要实现相应的接口

  5.编写完成后在web.xml文件中配置一下,就可以起作用了

  6.可以在不修改现有系统基础上,增加web应用程序生命周期事件的跟踪

servlet 规范中为每种事件监听器都定义了相应的接口,在编写事件监听器程序时只需实现这些接口就可以了。一些Servlet事件监听器需要在web应用程序的部署 文件描述符文件(web.xml)中进行注册(注册之后才能发布),一个web.xml可以注册多个servlet事件监听器。web服务器按照它们在web.xml中注册顺序来加载和注册这些servlet事件监听器。servlet事件监听器的注册和调用过程都是由web容器自动完成的,当发生被监听对象被创建,修改,销毁等事件时,web容器将调用与之相关的servlet事件监听器对象的相应方法(所监听到的对象如果在创建、修改、销毁事件触发的时候就会调用这些监听器这就相当于面向事件编程的概念),用户在这些方法中编写的事件处理代码(相当于JS中的事件响应)即被执行。由于在一个web应用程序中只会为每个事件监听器类创建一个实例对象,有可能出现多个线程同时调用一个事件监听对象的情况,所以要注意多线程安全问题。

二、监听器类型

按监听的对象划分:servlet2.4规范定义的事件有三种:

1.用于监听应用程序环境对象(ServletContext)的事件监听器

2.用于监听用户会话对象(HttpSession)的事件监听器

3.用于监听请求消息对象(ServletRequest)的事件监听器

 

按监听的事件类项划分

1.用于监听域对象自身的创建和销毁的事件监听器

2.用于监听域对象中的属性的增加和删除的事件监听器

3.用于监听绑定到HttpSession域中的某个对象的状态的事件监听器

 

在一个web应用程序的整个运行周期内,web容器会创建和销毁三个重要的对象,ServletContext,HttpSession,ServletRequest。

 

PS:其中Context 为JSP页面包装页面的上下文.由容器创建和初始化,管理对属于JSP中特殊可见部分中已命名对象的访问. 该接口用来定义了一个Servlet的环境对象。也可认为这是多个客户端共享的信息,它与session的区别在于应用范围的不同,session只对应于一个用户。 

servlet2.4中定义了三个接口:

ServletContextListener,HttpSessionListener,ServletRequestListener。分别实现对应的接口就可以实现对应的监听处理

在ServletContextListener接口中定义了两个事件处理方法,分别是

 

contextInitialized()和contextDestroyed()

public void contextInitialized(ServletcontextEvent sce)

这个方法接受一个ServletContextEvent类型参数,在contextInitialized可以通过这个参数获得当前被创建的ServletContext对象。

public void contextDestroyed(ServletContextEvent sce)

2.在HttpSessionListneter接口中共定义了两个事件处理方法,分别是sessionCreated()和sessionDestroyed()

public void sessionCreated(HttpSessionEvent se)

这个方法接受一个(HttpSessionEvent 类型参数,在sessionCreated可以通过这个参数获得当前被创建的HttpSession对象。

public void sessionDestroyed(HttpSessionEvent se)

 3.在ServletRequestListener接口中定义了两个事件处理方法,分别是requestInitialized()和requestDestroyed()

public void requestInitialized(ServletRequestEvent sre)

这个方法接受一个(ServletRequestEvent 类型参数,在requestInitialized可以通过这个参数获得当前被创建的ServletRequest对象。

public void requestDestroyed(ServletRequestEvent sre)

可 以看出三个监听器接口中定义的方法非常相似,执行原理与应用方式也相似,在web应用程序中可以注册一个或者多个实现了某一接口的事件监听器,web容器 在创建或销毁某一对象(如ServletContext,HttpSession)时就会产生相应的事件对象

(如ServletcontextEvent ,或者HttpSessionEvent),接着依次调用每个事件监听器中的相应处理方法,并将产生的事件对象传递给这些方法。

三、分类及介绍

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

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

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

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

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

四、部署

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

五、示例

第一步:编写监听器类

[html] view plaincopy
print?
  1. package cn.listen;  
  2.   
  3. import javax.servlet.ServletContextEvent;  
  4.   
  5. import javax.servlet.ServletContextListener;  
  6.   
  7.   
  8. public class MyListener implements ServletContextListener {  
  9.   
  10.     public void contextDestroyed(ServletContextEvent sce) {  
  11.   
  12.            System.out.println("die");  
  13.   
  14.     }  
  15.   
  16.     public void contextInitialized(ServletContextEvent sce) {  
  17.   
  18.            System.out.println("init");   
  19.   
  20.     }  
  21.   
  22. }   


第二步:布置安装

<listener> 

  <listener-class>cn.listen.MyListener</listener-class>

</listener> 

运行服务器会出现

[20:42:38.406] {main} WebApp[http://default] active

 init

[20:42:38.437] {main} WebApp[http://default/MyProj] active 

监听到了应用启动。


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.yaotu.net/news/605602.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Java Web中的EL(表达式语言)详解

Java Web中的EL&#xff08;表达式语言&#xff09;详解 表达式语言(Expression Language)简称EL&#xff0c;它是JSP2.0中引入的一个新内容。通过EL可以简化在JSP开发中对对象的引用&#xff0c;从而规范页面代码&#xff0c;增加程序的可读性及可维护性。EL为不熟悉Java语言页…

PowerDesigner入门教程

PowerDesigner安装方法:http://dev.firnow.com/course/3_program/java/javajs/20090908/174375.html安装完这2个软件了,接着就开始使用吧,还是和以前一样,先提出需求,再做实例,因为我们之所以使用它们,肯定是因为它们能巧妙的解决某种问题,不然我们干嘛花那么多时间和硬盘空间安…

PowerDesigner(一)-PowerDesigner概述(系统分析与建模)

PowerDesigner概述 PowerDesigner是Sybase公司推出的一个集成了企业架构&#xff0c;UML(统一建模语言)和数据库的CASE(计算机辅助软件工程)工具。它不仅可以用于系统设计和开发的不同阶段(即业务分析&#xff0c;概念模型设计&#xff0c;物理模型设计以及面向对象开发阶段)&a…

PowerDesigner(二)-项目和框架矩阵

项目和框架矩阵 项目是PowerDesigner 15的新概念&#xff0c;通过项目系统分析/设计人员可以对模型以及各类文档进行分组。项目也可以包含框架矩阵&#xff0c;以表格的形式体现各个模型之间的关系。 项目和框架矩阵解决了如何对模型进行统一管理的问题。 1.创建框架矩阵(FEAF-…

PowerDesigner(三)-企业架构模型

企业架构模型(Enterprise Architecture Model,EAM)是PowerDesigner 15新增的功能&#xff0c;它能够以图形的方式展现企业架构&#xff0c;从而取代文字描述&#xff1b;以偏向非技术性的表达方式&#xff0c;从不同层面表达不同的图示结果。 企业架构模型从业务层&#xff0c;…

PowerDesigner(四)-业务处理模型

业务处理模型 业务处理模型(Business Process Model,BPM)以业务需求作为出发点&#xff0c;用图形的方式描述系统的任务和业务流程&#xff0c;注重的是处理过程中数据流程。业务处理模型是从业务人员的角度对业务逻辑和规则进行详细描述的概念模型&#xff0c;并使用流程 图表…

PowerDesigner(五)-概念数据模型(CDM生成LDM,PDM和OOM)

概念数据模型 概念数据模型(Conceptual Data Model,CDM)&#xff1a;表达的是数据整体逻辑结构&#xff0c;该结构独立于任何软件和数据存储结构&#xff0c;即它只是系统分析人员&#xff0c;应用程序设计人员&#xff0c;维护人员和用户之间相互理解的共同语言&#xff0c;并…

PowerDesigner(六)-物理数据模型(PDM逆向工程)

物理数据模型PDM 物理数据模型(Physical Data Model,PDM):在数据库的逻辑结构设计好之后&#xff0c;就需要完成其物理设计&#xff0c;PDM就是为实现这一目的而设计的。 物理数据模型是以常用的DBMS(数据库管理系统)理论为基础&#xff0c;将CDM/LDM中所建立的现实世界模型生…

PowerDesigner(八)-面向对象模型(用例图,序列图,类图,生成Java源代码及Java源代码生成类图)

面向对象模型 面向对象模型是利用UML(统一建模语言)的图形来描述系统结构的模型&#xff0c;它从不同角度实现系统的工作状态。这些图形有助于用户&#xff0c;管理人员&#xff0c;系统分析人员&#xff0c;开发人员&#xff0c;测试人员和其他人员之间进行信息交流。这里主要…

PowerDesigner(九)-模型文档编辑器(生成项目文档)

模型文档编辑器 PowerDesigner的模型文档(Model Report)是基于模型的&#xff0c;面向项目的概览文档&#xff0c;提供了灵活&#xff0c;丰富的模型文档编辑界面&#xff0c;实现了设计&#xff0c;修改和输出模型文档的全过程。 模型文档的功能如下&#xff1a; 为各个模型…

JNDI学习总结(一)——JNDI数据源的配置

一、数据源的由来 在Java开发中&#xff0c;使用JDBC操作数据库的四个步骤如下&#xff1a; ①加载数据库驱动程序(Class.forName("数据库驱动类");)②连接数据库(Connection con DriverManager.getConnection();)③操作数据库(PreparedStatement stat con.prepa…

JNDI学习总结(二)——Tomcat下使用C3P0配置JNDI数据源

一、C3P0下载 C3P0下载地址&#xff1a;http://sourceforge.net/projects/c3p0/files/?sourcenavbar 下载完成之后得到一个压缩包。 二、使用C3P0配置JNDI数据源 Tomcat6.x中配置JNDI数据源时默认使用的是Tomcat6.x自带的DBCP连接池&#xff0c;Tomcat6.x使用DBCP连接池配置JN…

java 消息机制 ActiveMQ入门实例

1.下载ActiveMQ 去官方网站下载&#xff1a; http://activemq.apache.org/我下载的时候是 ActiveMQ 5.8.0 Release版 2.运行ActiveMQ 解压缩apache-activemq-5.8.0-bin.zip&#xff0c;然后双击apache-activemq-5.5.1\bin\activemq.bat运行ActiveMQ程序。 启动ActiveMQ以后&…

JavaSE基础12笔记常见对象

12.01_常见对象(Scanner概述及其方法介绍) A.Scanner概述 B.Scanner构造方法原理 ​ Scanner(Inputstream source) ​ System下面有个字段 ​ public static final Inputstream in;标准输入流,对应键盘录入 C.一般方法 ​ hasNextxxx() 判断是否还有下一个输入项 ​ ne…

spring integeration主要概念

1目标和原则 SpringIntegration主要有如下两个目标&#xff1a; l提供一个简单的模型来实现复杂的企业集成解决方案 l为基于Spring的应用添加异步的、消息驱动的行为&#xff0c;让Spring用户可以直观的、增量的采用 SpringIntegration基于两个原则&#xff1a; l组件应该松耦合…

Spring Integration入门

为什么使用Spring Integration Spring Integration是Spring框架创建的又一个API&#xff0c;面向企业应用集成&#xff08;EAI&#xff09;。说到集成&#xff0c;并不缺“解决办法”&#xff1a;硬编码的Java客户端、其它ESB产品&#xff0c;还有消息队列等更加传统的应用集成…

dispatcherservlet详解

3.1、DispatcherServlet作用 DispatcherServlet是前端控制器设计模式的实现&#xff0c;提供Spring Web MVC的集中访问点&#xff0c;而且负责职责的分派&#xff0c;而且与Spring IoC容器无缝集成&#xff0c;从而可以获得Spring的所有好处。 具体请参考第二章的图2-1。 Dispa…

理解本真的REST架构风格

引子 在移动互联网、云计算迅猛发展的今天&#xff0c;作为一名Web开发者&#xff0c;如果您还没听说过“REST”这个buzzword&#xff0c;显然已经落伍了。夸张点说&#xff0c;甚至“出了门都不好意思跟别人打招呼”。尽管如此&#xff0c;对于REST这个泊来品的理解&#xff0…

JavaSE基础21笔记IO流

22.01_IO流(序列流)(了解) 1.什么是序列流 序列流可以把多个字节输入流整合成一个, 从序列流中读取数据时, 将从被整合的第一个流开始读, 读完一个之后继续读第二个, 以此类推. 2.使用方式 整合两个: SequenceInputStream(InputStream, InputStream) FileInputStream fis1 n…

深入浅出REST

不知你是否意识到&#xff0c;围绕着什么才是实现异构的应用到应用通信的“正确”方式&#xff0c;一场争论正进行的如火如荼&#xff1a;虽然当前主流的方式明显地集中在基于SOAP、WSDL和WS-*规范的Web Services领域&#xff0c;但也有少数人用细小但洪亮的声音主张说更好的方…
最新文章