Java servlet、filter、listener、interceptor之间的区别和联系
作者:lqh 发布时间:2023-11-02 15:32:17
servlet、filter、listener、interceptor之间的区别和联系
一、概念
1.servlet:servlet是一种运行服务器端的java应用程序,具有独立于平台和协议的特性,并且可以动态的生成web页面,它工作在客户端请求与服务器响应的中间层。
2.filter:filter是一个可以复用的代码片段,可以用来转换HTTP请求、响应和头信息。Filter不像Servlet,它不能产生一个请求或者响应,它只是修改对某一资源的请求,或者修改从某一的响应。
3.listener: * ,从字面上可以看出listener主要用来监听只用。通过listener可以监听web服务器中某一个执行动作,并根据其要求作出相应的响应。
通俗的语言说就是在application,session,request三个对象创建消亡或者往其中添加修改删除属性时自动执行代码的功能组件。
4.interceptor:是在面向切面编程的,就是在你的service或者一个方法,前调用一个方法,或者在方法后调用一个方法。
比如 * 就是 * 的简单实现,在你调用方法前打印出字符串(或者做其它业务逻辑的操作),也可以在你调用方法后打印出字符串,甚至在你抛出异常的时候做业务逻辑的操作。
5.servlet、filter、listener是配置到web.xml中,interceptor不配置到web.xml中,struts的 * 配置到struts.xml中。spring的 * 配置到spring.xml中。
二、加载顺序
web.xml 的加载顺序是:context- param -> listener -> filter -> servlet
三、职责
1.servlet:
(1)创建并返回一个包含基于客户请求性质的动态内容的完整的html页面
(2)创建可嵌入到现有的html页面中的一部分html页面(html片段)
(3)读取客户端发来的隐藏数据
(4)读取客户端发来的显示数据
(5)与其他服务器资源(包括数据库和java的应用程序)进行通信
(6)通过状态代码和响应头向客户端发送隐藏数据。
2.filter:
(1)filter能够在一个请求到达servlet之前预处理用户请求,也可以在离开servlet时处理http响应
(2)在执行servlet之前,首先执行filter程序,并为之做一些预处理工作
(3)根据程序需要修改请求和响应
(4)在servlet被调用之后截获servlet的执行。
3.listener:
servlet2.4规范中提供了8个 listener 接口,可以将其分为三类,分别如下:
(1)与 servletContext 有关的 listne r接口。包括:ServletContextListener、ServletContextAttributeListener
(2)与 HttpSession 有关的 Listner 接口。包括:HttpSessionListner、HttpSessionAttributeListener、HttpSessionBindingListener、 HttpSessionActivationListener
(3)与 ServletRequest 有关的 Listener 接口,包括:ServletRequestListner、ServletRequestAttributeListener
四、区别
1.servlet:servlet 流程是短的,url传来之后,就对其进行处理,之后返回或转向到某一自己指定的页面。它主要用来在业务处理之前进行控制。
2.filter:流程是线程性的,url传来之后,检查之后,可保持原来的流程继续向下执行,被下一个filter, servlet接收等,而 servlet 处理之后,不会继续向下传递。
filter 功能可用来保持流程继续按照原来的方式进行下去,或者主导流程,而servlet的功能主要用来主导流程。可以将 Filter 看成是 servlet 的一个补充(擦屁股的)。
Filter可认为是Servlet的一种“变种”,它主要用于对用户请求进行预处理,也可以对HttpServletResponse进行后处理,是个典型的处理链。
它与Servlet的区别在于:它不能直接向用户生成响应。
完整的流程是:Filter对用户请求进行预处理,接着将请求交给Servlet进行处理并生成响应,最后Filter再对服务器响应进行后处理。
3.匹配规则
当一个请求发送到servlet容器的时候,容器先会将请求的url减去当前应用上下文的路径作为servlet的映射url,比如我访问的是http://localhost/test/aaa.html(我的应用上下文是test),
容器会将http://localhost/tes去掉,将剩下的/aaa.html部分拿来做servlet的映射匹配,也就是拿这剩下的部分与web.xml中配置的servlet的url-pattern进行匹配。
注意:这个映射匹配过程是有一定的规则的,而且每次匹配最终都只匹配一个 servlet。(这一点和filter不同)
servlet 匹配规则:当一個servlet匹配成功后就不会在往下去匹配了
精确路径的匹配:
例子:比如servletA 的url-pattern为 /test,servletB的url- pattern为 /* ,这个时候,如果我访问的url为http://localhost/test ,
这个时候容器就会先 进行精确 路径匹配,发现/test正好被servletA精确匹配,那么就去调用servletA,也不会去理会其他的servlet了。
最长路径的匹配:
例子:servletA的url-pattern为/test/*,而servletB的url-pattern为/test/a/*,此时访问http://localhost/test/a时,
容器会选择路径最长的servlet来匹配,也就是这里的servletB。
扩展匹配:如果url最后一段包含扩展,容器将会根据扩展选择合适的servlet。
例子:servletA的url-pattern:*.action
4.servlet,filter 都是针对 url 之类的,而 listener 是针对对象的操作的,如 session 的创建,session.setAttribute 的发生,在这样的事件发生时做一些事情。
可用来进行:Spring整合Struts,为Struts的action注入属性,web应用定时任务的实现,在线人数的统计等
5.interceptor * ,类似于filter,不过在struts.xml中配置,不是在web.xml,并且不是针对URL的,而是针对action,当页面提交action时,
进行过滤操作,相当于struts1.x提供的plug-in机制,可以看作,前者是struts1.x自带的filter,而interceptor 是struts2 提供的filter。
与filter不同点:
(1)不在web.xml中配置,而是在struts.xml中完成配置,与action在一起
(2)可由action自己指定用哪个interceptor 来在接收之前做事
6.struts2中的过滤器和 * 的区别与联系:
(1) * 是 Struts2 提供的,而过滤器是由 Servlet 标准提供的
(2) * 拦截目标 Action 的目标方法,而过滤器针对各种 web 资源
(3) * 在 struts.xml 中配置,而过滤器在 web.xml文件中配置
(4) * 使用 * 栈组织在一起,而过滤器是根据被拦截的资源联系在一起,由他们在配置文件中的位置决定了先后执行顺序
(5) * 是基于java反射机制的,而过滤器是基于函数回调的。
(6)过滤器依赖与servlet容器,而 * 不依赖与servlet容器。
(7) * 只能对Action请求起作用,而过滤器则可以对几乎所有请求起作用。
(8) * 可以访问Action上下文、值栈里的对象,而过滤器不能。
(9)在Action的生命周期中, * 可以多次调用,而过滤器只能在容器初始化时被调用一次。
感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!


猜你喜欢
- 打注解@SpringBootTest的时候不会出现提示但是又导入了 <dependency> &nb
- 如果你还不是很了解restful,或者认为restful只是一种规范不具有实际意义,推荐一篇osc两年前的文章:RESTful API 设计
- 前言在上一篇文章中讲完了SpringMVC的大部分知识,此篇文章中主要讲解 * 。上一篇文章🚩 * 的使用是非常普遍的。例如在 OA系统中通
- volatile对 volatile的理解volatile 是一种轻量级的同步机制。保证数据可见性不保证原子性禁止指令重排序JMMJMM(J
- 前言在Android屏幕的空间中,大部分的区域我们都是可以随意绘制,只有一部分区域是显示的固定内容:状态栏标题栏(ActionBar)页面内
- 限流器算法目前常用限流器算法为两种:令牌桶算法和漏桶算法,主要区别在于:漏桶算法能够强行限制请求速率,平滑突发请求,而令牌桶算法在限定平均速
- 需求描述现在有这样一个需求:我有A、B两台服务器,其中A是一个视频处理服务器,B是一个数据存储服务器。此时有一个视频需要先在A服务器上进行一
- /// <summary> /// 计算本周起始日期(礼拜一的日期) /// </summary&
- IO感觉上和多线程并没有多大关系,但是NIO改变了线程在应用层面使用的方式,也解决了一些实际的困难。而AIO是异步IO和前面的系列也有点关系
- 本文实例为大家分享了C#点餐系统的具体代码,供大家参考,具体内容如下using System;using System.Collection
- 前言单例模式,是工作中比较常见的一种设计模式,通常有两种实现方式,懒汉式和饿汉式。但是这两种实现方式存在一些问题。懒汉式需要在多线程环境下使
- 多说无益,贴代码:/** * 校验银行卡卡号 * * @param cardId &nbs
- 前言此文章主要解决拦截用户点击手机底部导航栏中的返回键时该事件的拦截;此方法依然可以适用于fragmentonBackPressed()这是
- Java中代码的加载顺序所能了解的知识点类的依赖关系static代码块的加载时间继承类中构造器的隐式调用非static的成员变量初始化时间m
- 碎片的创建要使用碎片先要创建一个碎片,创建一个碎片很简单。1.新建一个碎片布局,fragment.xml<?xml version=&
- Java 序列化和反序列化实例详解在分布式应用中,对象只有经过序列化才能在各个分布式组件之间传输,这就涉及到两个方面的技术-发送者将对象序列
- 异常是一个事件,它发生在程序运行期间,干扰了正常的指令流程。Java通过API中Throwable类的众多子类描述各种不同的异常。因而,Ja
- 先来看我们以前利用RestTemplate发起远程调用的代码:存在下面的问题:代码可读性差,编程体验不统一参数复杂URL难以维护1. Fei
- 重载1.构造器的重载因为构造器的名字必须与类名相同,所以同一个类的所有构造器名肯定相同,构成重载;为了让系统能区分不同的构造器,多个构造器的
- 本文开始做一个网上商城的项目,首先从搭建环境开始,一步步