[Java Web]Filter | 一文搞懂Web三大组件之一的Filter

⭐作者介绍:大二本科网络工程专业在读,持续学习Java,努力输出优质文章
⭐作者主页:@逐梦苍穹
⭐所属专栏:Java Web

写在前面:本文介绍Web三大组件之一的Filter过滤器,另外两个组件为Servlet和Listener。
Servlet链接: Servlet
Listener链接:[Java Web]Listener | 一文带你上手Web三大组件之一的listener

Filter

1、概述

  Filter 表示过滤器,是 JavaWeb 三大组件(Servlet、Filter、Listener)之一。
 在Java Web开发中,Filter是一种常用的组件,它可以在请求到达Web应用程序的Servlet之前或之后对请求进行拦截和处理
 Filter可以用来对请求参数进行过滤、对响应结果进行加工处理、实现安全认证、记录日志等等。

 过滤器可以把对资源的请求拦截下来,从而实现一些特殊的功能。

如下图所示,浏览器可以访问服务器上的所有的资源(servlet、jsp、html等)

而在访问到这些资源之前可以使过滤器拦截来下,也就是说在访问资源之前会先经过 Filter,如下图

拦截器拦截到后可以做什么?
过滤器一般完成一些通用的操作。比如每个资源都要写一些代码完成某个功能,我们不可能在每一个部分都编写大量重复的代码,而此时可以将这些代码写在过滤器中,因为请求每一个资源都要经过过滤器。

2、执行流程

2.1、简单介绍

如上图是使用过滤器的流程,那么便有如下问题:
● 放行后访问对应资源,资源访问完成后,还会回到Filter中吗?从上图就可以看出肯定会回到Filter中
● 如果回到Filter中,是重头执行还是执行放行后的逻辑呢?如果是重头执行,就意味着 放行前逻辑 会被执行两次,肯定不符合要求;
所以访问完资源后,会回到 放行后逻辑,执行该部分代码。

对于过滤器的执行流程,归结起来就是:
 执行放行前逻辑->放行>访问相应的资源>执行放行后的逻辑

2.2、过滤器链

 过滤器链(Filter Chain)是由多个Filter组成的链式结构,用于对HTTP请求和响应进行拦截和处理。
 当一个请求到达Servlet容器时,Servlet容器会按照在web.xml中配置的顺序依次调用过滤器链中的每一个Filter,直到请求被Servlet处理或是Filter链被中断。

简单来说:过滤器链是指在一个Web应用,可以配置多个过滤器,这多个过滤器称为过滤器链。

过滤器链执行流程:

  1. 执行 Filter1 的放行前逻辑代码
  2. 执行 Filter1 的放行代码
  3. 执行 Filter2 的放行前逻辑代码
  4. 执行 Filter2 的放行代码
  5. 访问到资源
  6. 执行 Filter2 的放行后逻辑代码
  7. 执行 Filter1 的放行后逻辑代码
    以上流程串起来就像一条链子,绕一圈回来,故称之为过滤器链。

那么就有另外一个问题,要怎么判断是哪个过滤器先执行?
 后面会提到过滤器具体代码怎么实现,使用的是注解配置的方式。
 这种方式的优先级是按照过滤器类名的字符串的自然排序。
举个例子:
 BFilterDemo和AFilterDemo,那就是AFilterDemo先执行。
 FilterDemo1和FilterDemo2那就肯定是FilterDemo1先执行。
挨个比对字符串。

3、代码实现

3.1、实现步骤

使用的步骤分为三步:

  1. 创建一个实现javax.servlet.Filter接口的Java类,并重写所有的方法
  2. 在重写的doFilter()方法中,可以对请求进行拦截和处理。如果需要将请求传递给下一个Filter或Servlet,则需要调用FilterChain对象的doFilter()方法。
  3. 配置拦截路径。有两种方式:
    3.1. 在web.xml中配置Filter,将Filter映射到需要拦截的URL地址上。
    3.2. 使用注解配置Filter拦截路径

 Java中的Filter接口定义了三个方法:init()、doFilter()和destroy(),分别用于初始化Filter、处理请求和销毁Filter。其中,doFilter()方法是Filter的核心方法,用于对请求进行拦截和处理。

3.2、🔺配置拦截路径的方式

在Java Web中,拦截路径的书写方式可以分为以下几种:

  1. 精确匹配:拦截路径可以直接指定需要拦截的URL路径,例如:“/admin/index.jsp”。此时,只有当请求的URL路径和拦截路径完全匹配时,Filter才会拦截该请求。
  2. 目录匹配:拦截路径可以以"/“结尾,表示拦截指定目录下的所有请求,例如:”/admin/“。此时,所有以”/admin/"开头的URL路径都会被拦截。
  3. 扩展名匹配:拦截路径可以以".“开头,表示拦截指定扩展名的请求,例如:”.jsp"。此时,所有以".jsp"结尾的URL路径都会被拦截。
  4. 多重匹配:拦截路径可以使用多重匹配的方式,例如:“/admin//.jsp”。此时,只有当URL路径以"/admin/“开头,并且含有至少一个”/“,且以”.jsp"结尾时,Filter才会拦截该请求。
  5. 正则表达式匹配:拦截路径可以使用正则表达式来指定需要拦截的URL路径。在使用正则表达式时,需要将拦截路径设置为"/*"(拦截所有),并在Filter中编写正则表达式的匹配规则。

3.3、具体实现

下面是具体实现:
写一个JSP文件:

访问一下,看看效果:

  是可以成功访问的。

下面编写过滤器代码:

package com.xzl.web;

import javax.servlet.*;
import javax.servlet.annotation.WebFilter;
import java.io.IOException;

/**
 * @author 逐梦苍穹
 * @date 2023/4/1 15:55
 */
@WebFilter("/*")
public class FilterDemo implements Filter {
    @Override
    public void init(FilterConfig filterConfig) throws ServletException {
        Filter.super.init(filterConfig);
    }

    @Override
    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
        System.out.println("doFilter..成功拦截");
    }

    @Override
    public void destroy() {
        Filter.super.destroy();
    }
}

这里配置拦截路径为拦截所有,下面再次访问JSP:
页面什么都没有,查看控制台:

3.4、🔺过滤器链实现

把刚刚创建的FilterDemo改为FilterDemo1,然后再创建一个FilterDemo2。
下面编写代码:

在控制台可以看到:

说明过滤器链的执行流程跟之前提到的是一样的:

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
好的,以下是将Java Web过滤器(Filter)示例中的所有请求参数转换为大写的示例代码: ```java import javax.servlet.*; import javax.servlet.annotation.WebFilter; import javax.servlet.http.HttpServletRequest; import java.io.IOException; @WebFilter(filterName = "toUpperCaseFilter", urlPatterns = "/*") public class ToUpperCaseFilter implements Filter { @Override public void init(FilterConfig filterConfig) throws ServletException { // do nothing } @Override public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException { HttpServletRequest request = (HttpServletRequest) servletRequest; ToUpperCaseRequestWrapper requestWrapper = new ToUpperCaseRequestWrapper(request); filterChain.doFilter(requestWrapper, servletResponse); } @Override public void destroy() { // do nothing } static class ToUpperCaseRequestWrapper extends HttpServletRequestWrapper { public ToUpperCaseRequestWrapper(HttpServletRequest request) { super(request); } @Override public String getParameter(String name) { String value = super.getParameter(name); if (value != null) { value = value.toUpperCase(); } return value; } @Override public String[] getParameterValues(String name) { String[] values = super.getParameterValues(name); if (values != null) { for (int i = 0; i < values.length; i++) { values[i] = values[i].toUpperCase(); } } return values; } } } ``` 以上代码中,我们创建了一个名为 `toUpperCaseFilter` 的过滤器,它会对所有的请求进行过滤。在过滤器中,我们将请求对象(request)转换为 `ToUpperCaseRequestWrapper` 对象,并将其传递给过滤器链中的下一个过滤器或目标组件。在 `ToUpperCaseRequestWrapper` 中,我们重写了 `getParameter` 和 `getParameterValues` 方法,在这些方法中将请求参数的值转换为大写。最后,我们将新的请求对象返回给过滤器链中的下一个组件。这样,所有的请求参数都会被转换为大写。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

逐梦苍穹

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值