热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Filter过滤器(1)

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

   Filter也称之为过滤器,它是Servlet技术中比较激动人心的技术,WEB开发人员通过Filter技术,对web服务器管理的所有web资源:例如Jsp, Servlet, 静态图片文件或静态 html 文件等进行拦截,从而实现一些特殊的功能。例如实现URL级别的权限访问控制、过滤敏感词汇、压缩响应信息等一些高级功能。

  其顺序图如下(Servlet API中提供了一个Filter接口,开发web应用时,如果编写的Java类实现了这个接口,则把这个java类称之为过滤器Filter。通过Filter技术,开发人员可以实现用户在访问某个目标资源之前,对访问的请求和响应进行拦截,简单说,就是可以实现web容器对某资源的访问前截获进行相关的处理,还可以在某资源向web容器返回响应前进行截获进行处理。)

技术分享

过滤器,设计执行流程:

  1. 用户访问服务器
  2. 过滤器: 对Servlet请求进行拦截
  3. 先进入过滤器, 过滤器处理
  4. 过滤器处理完后, 在放行, 此时,请求到达Servlet/JSP
  5. Servlet处理
  6. Servlet处理完后,再回到过滤器, 最后在由tomcat服务器相应用户;

Filter的生命周期

(1)、init(FilterConfig filterConfig)throws ServletException:

  与servlet一样(Filter其实也是一个servlet),Filter的创建和销毁是由web服务器负责的。web服务器一启动,就会调用init()方法,完成对象的初始化功能

  (注:filter对象只会创建一次,init方法也只会执行一次。)

(2)、destroy():

  在Web容器卸载 Filter 对象之前被调用。该方法在Filter的生命周期中仅执行一次。在这个方法中,可以释放过滤器使用的资源。

(3)、doFilter(ServletRequest request, ServletResponse response, FilterChain chain)

   The doFilter method of the Filter is called by the container each time a request/response pair is passed through the chain due to a client request for a resource at the end of the chain.(大概的意思是在每次请求/响应时由容器创建)过滤器业务拦截的业务处理方法


程序示例

首先写一个要被访问的servlet,如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
package com.gqx.filter;
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 IndexServlet extends HttpServlet {
    /**
     * 处理用户http request
     */
    private static final long serialVersiOnUID= 1L;
    public void doGet(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {
        System.out.println("4、处理用户请求开始!");
    }
    public void doPost(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {
        this.doGet(request, response);
    }
}

然后就是去访问之前需要对请求或响应内容做处理的过滤器

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
package com.gqx.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;
/**
 * 测试的过滤器
 * @author Administrator
 *
 */
public class HelloFilter implements Filter {
    //创建实例
    public HelloFilter() {
    // TODO Auto-generated constructor stub
        System.out.println("1、过滤器用例开始创建了。");
    }
    @Override
    public void init(FilterConfig filterConfig) throws ServletException {
        // TODO Auto-generated method stub
        System.out.println("2、执行过滤器初始化方法!");
    }
    //过滤器业务处理方法,在请求到达servlet之前进入此方法处理公用的业务逻辑操作
    @Override
    public void doFilter(ServletRequest request, ServletResponse response,
            FilterChain chain) throws IOException, ServletException {
        // TODO Auto-generated method stub
        System.out.println("3、执行过滤器的业务处理方法!");
        //放行(去到Servlet)
        chain.doFilter(request, response);
     
        System.out.println("5、Servlet处理完成,又回到Filter了!");
    }
    @Override
    public void destroy() {
        // TODO Auto-generated method stub
        System.out.println("6、同时过滤器被销毁了!");
    }
}

最后就是要去web.xml中去配置好先关的过滤器设置

1
2
3
4
5
6
7
8
9
10
11
12
13
    
    
    HelloFilter
    
    class>com.gqx.filter.HelloFilterclass>
    
    HelloFilter
    
    /*

一切都ok了现在我在各个打印语句之前都设置了断点,可以观察到如下图所示的变化

技术分享


当有多个过滤器的时候,如下

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
    
        
        HelloFilter
        
        class>com.gqx.filter.HelloFilterclass>
    
    
        
        HelloFilter
        
        /*
    
     
    
    
        HelloFiler2
        class>com.gqx.filter.HelloFilter2class>
    
    
        HelloFiler2
        /*
    

同理在第一个HelloFilter的doFilter()的方法中写入

1
2
3
4
5
  System.out.println("3、第一个过滤器:执行过滤器的业务处理方法!");
//放行(去到Servlet)
chain.doFilter(request, response);
System.out.println("5、第一个过滤器:Servlet处理完成,又回到Filter了!");

在HelloFiler2的doFilter()的方法中写入

1
2
3
System.out.println("第二个过滤器!");
        chain.doFilter(request, response);
        System.out.println("第二个过滤器结束了!");

 同样在这些语句中设置断点会发现如下结果

技术分享

这个时候就可以知道服务器启动后,用户去访问,根据xml中的过滤器的先后顺序来执行对应的过滤器,执行其对应的doFilter的方法放行request/response,被放行后他们就进入了第二个过滤器HelloFiler2,被第二个过滤器放行后才会回到对应的servlet,servlet运行完之后,回到第二个过滤器,再回到第一个过滤器。(如下图)

 技术分享


过滤器相关Api

(1)、|-- interface  Filter                            过滤器核心接口

         Void  init(filterConfig);    初始化方法,在服务器启动时候执行

   Void  doFilter(request,response,filterChain);   过滤器拦截的业务处理方法

   Void destroy();                         销毁过滤器实例时候调用

(2)|-- interface  FilterConfig   获取初始化参数信息(方法类似于servletConfig)

  String getInitParameter(java.lang.String name)

  Enumeration getInitParameterNames() 


与servlet同样,在web.xml中配置先关信息

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
    
    
    
        encoding
        UTF-8
    
    
        path
        c:/...
    
    FilterApi
    class>com.gqx.filter.FilterApiclass>
    FilterApi
    /*
   

(注意:有时候会出现错误 cvc-complex-type.2.4.a: Invalid content was found starting with element 这个时候可以将“http://www.springmodules.org/schema/cache/springmodules-cache.xsd http://www.springmodules.org/schema/cache/springmodules-ehcache.xsd”这段话加入到xml文件的"xmlns:xsi="的标签中)

如下:

1 "http://www.w3.org/2001/XMLSchema-instance http://www.springmodules.org/schema/cache/springmodules-cache.xsd http://www.springmodules.org/schema/cache/springmodules-ehcache.xsd" xmlns="http://java.sun.com/xml/ns/javaee" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd" id="WebApp_ID" version="3.0">

然后就是他的过滤器先关的内容了

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
public class FilterApi implements Filter {
    @Override
    public void init(FilterConfig filterConfig) throws ServletException {
        // TODO Auto-generated method stub
        System.out.println("获取过滤器在web.xml中配置的初始化参数!");
        String encoding=filterConfig.getInitParameter("encoding");
        System.out.println(encoding);
        //filterConfig.getFilterName() 获取单个
        Enumeration enums=filterConfig.getInitParameterNames();
        while (enums.hasMoreElements()) {
            //获取对应的参数名称
            String name = (String) enums.nextElement();
            //获取对应名称的值
            String value=filterConfig.getInitParameter(name);
            System.out.println(name+"\t"+value);
        }
    }
    @Override
    public void destroy() {
        // TODO Auto-generated method stub
    }
    @Override
    public void doFilter(ServletRequest request, ServletResponse response,
            FilterChain chain) throws IOException, ServletException {
        // TODO Auto-generated method stub 
    }
}

启动服务器就会看到控制台打印的先关数据

技术分享

(3)、|-- interface  FilterChain     过滤器链参数;一个个过滤器形成一个执行链;

         void doFilter(ServletRequest request, ServletResponse response)  ;  执行下一个过滤器或放行


对指定的请求拦截

刚才看到的是对所有的请求进行拦截

1
2
3
4
  
    FilterApi
    /*
1
2
3
4
5
6
7
@Override
public void doFilter(ServletRequest request, ServletResponse response,
        FilterChain chain) throws IOException, ServletException {
    // TODO Auto-generated method stub
    System.out.println("指定过滤器的拦截。。。。。");
    chain.doFilter(request, response);
}

当访问index.jsp的时候,就会出现

技术分享

当要对多个指定的请求进行拦截时,可以这样写

1
2
3
4
5
6
7
8
9
10
    
    PontedFilter
    /index.jsp
    /list.jsp
    
    
    /IndexServlet
     

规则有点像servlet的配置

对类型的拦截

1
2
3
4
5
 
REQUEST 
FORWARD 
INCLUDE 
ERROR

默认拦截的类型:(直接访问或者重定向)

REQUEST

拦截转发:

                     FORWARD

拦截包含的页面(RequestDispatcher.include(/page.jsp);    对page.jsp也执行拦截)

                     INCLUDE

拦截声明式异常信息:

                     ERROR

      类似于如下代码

1
2
3
4
5
   
    404
    
    

 如下代码示例

1
2
/*
FORWARD 

 在ServletTets中写下

1 request.getRequestDispatcher("IndexServlet").forward(request, response);

请求的转发就会被拦截住。

Filter过滤器(1)


推荐阅读
  • 本文介绍了OC学习笔记中的@property和@synthesize,包括属性的定义和合成的使用方法。通过示例代码详细讲解了@property和@synthesize的作用和用法。 ... [详细]
  • Mac OS 升级到11.2.2 Eclipse打不开了,报错Failed to create the Java Virtual Machine
    本文介绍了在Mac OS升级到11.2.2版本后,使用Eclipse打开时出现报错Failed to create the Java Virtual Machine的问题,并提供了解决方法。 ... [详细]
  • 在说Hibernate映射前,我们先来了解下对象关系映射ORM。ORM的实现思想就是将关系数据库中表的数据映射成对象,以对象的形式展现。这样开发人员就可以把对数据库的操作转化为对 ... [详细]
  • 本文介绍了在SpringBoot中集成thymeleaf前端模版的配置步骤,包括在application.properties配置文件中添加thymeleaf的配置信息,引入thymeleaf的jar包,以及创建PageController并添加index方法。 ... [详细]
  • 知识图谱——机器大脑中的知识库
    本文介绍了知识图谱在机器大脑中的应用,以及搜索引擎在知识图谱方面的发展。以谷歌知识图谱为例,说明了知识图谱的智能化特点。通过搜索引擎用户可以获取更加智能化的答案,如搜索关键词"Marie Curie",会得到居里夫人的详细信息以及与之相关的历史人物。知识图谱的出现引起了搜索引擎行业的变革,不仅美国的微软必应,中国的百度、搜狗等搜索引擎公司也纷纷推出了自己的知识图谱。 ... [详细]
  • 本文详细介绍了Linux中进程控制块PCBtask_struct结构体的结构和作用,包括进程状态、进程号、待处理信号、进程地址空间、调度标志、锁深度、基本时间片、调度策略以及内存管理信息等方面的内容。阅读本文可以更加深入地了解Linux进程管理的原理和机制。 ... [详细]
  • 1,关于死锁的理解死锁,我们可以简单的理解为是两个线程同时使用同一资源,两个线程又得不到相应的资源而造成永无相互等待的情况。 2,模拟死锁背景介绍:我们创建一个朋友 ... [详细]
  • 后台获取视图对应的字符串
    1.帮助类后台获取视图对应的字符串publicclassViewHelper{将View输出为字符串(注:不会执行对应的ac ... [详细]
  • 《数据结构》学习笔记3——串匹配算法性能评估
    本文主要讨论串匹配算法的性能评估,包括模式匹配、字符种类数量、算法复杂度等内容。通过借助C++中的头文件和库,可以实现对串的匹配操作。其中蛮力算法的复杂度为O(m*n),通过随机取出长度为m的子串作为模式P,在文本T中进行匹配,统计平均复杂度。对于成功和失败的匹配分别进行测试,分析其平均复杂度。详情请参考相关学习资源。 ... [详细]
  • Java验证码——kaptcha的使用配置及样式
    本文介绍了如何使用kaptcha库来实现Java验证码的配置和样式设置,包括pom.xml的依赖配置和web.xml中servlet的配置。 ... [详细]
  • CentOS 6.5安装VMware Tools及共享文件夹显示问题解决方法
    本文介绍了在CentOS 6.5上安装VMware Tools及解决共享文件夹显示问题的方法。包括清空CD/DVD使用的ISO镜像文件、创建挂载目录、改变光驱设备的读写权限等步骤。最后给出了拷贝解压VMware Tools的操作。 ... [详细]
  • VScode格式化文档换行或不换行的设置方法
    本文介绍了在VScode中设置格式化文档换行或不换行的方法,包括使用插件和修改settings.json文件的内容。详细步骤为:找到settings.json文件,将其中的代码替换为指定的代码。 ... [详细]
  • 本文介绍了使用Java实现大数乘法的分治算法,包括输入数据的处理、普通大数乘法的结果和Karatsuba大数乘法的结果。通过改变long类型可以适应不同范围的大数乘法计算。 ... [详细]
  • 本文介绍了Java工具类库Hutool,该工具包封装了对文件、流、加密解密、转码、正则、线程、XML等JDK方法的封装,并提供了各种Util工具类。同时,还介绍了Hutool的组件,包括动态代理、布隆过滤、缓存、定时任务等功能。该工具包可以简化Java代码,提高开发效率。 ... [详细]
  • web.py开发web 第八章 Formalchemy 服务端验证方法
    本文介绍了在web.py开发中使用Formalchemy进行服务端表单数据验证的方法。以User表单为例,详细说明了对各字段的验证要求,包括必填、长度限制、唯一性等。同时介绍了如何自定义验证方法来实现验证唯一性和两个密码是否相等的功能。该文提供了相关代码示例。 ... [详细]
author-avatar
为谁落慕
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有