1. 当前的OA项目存在什么缺陷? 56
- DeptServlet、EmpServlet、OrderServlet。每一个Servlet都是处理自己相关的业务。在这些Servlet执行之前都是需要判断用户是否登录了。如果用户登录了,可以继续操作,如果没有登录,需要用户登录。这段判断用户是否登录的代码是固定的,并且在每一个Servlet类当中都需要编写,显然代码没有得到重复利用。包括每一个Servlet都要解决中文乱码问题,也有公共的代码。这些代码目前都是重复编写,并没有达到复用。怎么解决这个问题?
- 可以使用Servlet规范中的Filter过滤器来解决这个问题。
这个改造我们写在上一课course26中
2. Filter是什么,有什么用,执行原理是什么? 56
- Filter是过滤器。
- Filter可以在Servlet这个目标程序执行之前添加代码。也可以在目标Servlet执行之后添加代码。之前之后都可以添加过滤规则。
- 一般情况下,都是在过滤器当中编写公共代码。
3. 一个过滤器怎么写呢? 56
3.1 第一步:
编写一个Java类实现一个接口:jarkata.servlet.Filter。并且实现这个接口当中所有的方法。
- init方法:在Filter对象第一次被创建之后调用,并且只调用一次。
- doFilter方法:只要用户发送一次请求,则执行一次。发送N次请求,则执行N次。在这个方法中编写过滤规则。
- destroy方法:在Filter对象被释放/销毁之前调用,并且只调用一次。
3.2 第二步:
在web.xml文件中对Filter进行配置。这个配置和Servlet很像。
<filter>
<filter-name>filter2</filter-name>
<filter-class>com.bjpowernode.javaweb.servlet.Filter2</filter-class>
</filter>
<filter-mapping>
<filter-name>filter2</filter-name>
<url-pattern>*.do</url-pattern>
</filter-mapping>
或者使用注解:@WebFilter({"*.do"})
代码在com.bjpowernode.javaweb.servlet
AServlet
package com.bjpowernode.javaweb.servlet;
import jakarta.servlet.ServletException;
import jakarta.servlet.annotation.WebServlet;
import jakarta.servlet.http.HttpServlet;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import java.io.IOException;
//Filter过滤器的介绍 56
@WebServlet("/a.do")
public class AServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest requset, HttpServletResponse response)
throws ServletException, IOException {
System.out.println("AServlet中的doGet方法执行了");
}
}
BServlet
package com.bjpowernode.javaweb.servlet;
import jakarta.servlet.ServletException;
import jakarta.servlet.annotation.WebServlet;
import jakarta.servlet.http.HttpServlet;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import java.io.IOException;
//Filter过滤器的介绍 56
@WebServlet("/b.do")
public class BServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest requset, HttpServletResponse response)
throws ServletException, IOException {
System.out.println("BServlet中的doGet方法执行了");
}
}
Filter1
package com.bjpowernode.javaweb.servlet;
import jakarta.servlet.*;
import jakarta.servlet.annotation.WebFilter;
import java.io.IOException;
//这是一个过滤器 56
//@WebFilter("/abc")
//@WebFilter("/a.do")
//@WebFilter({"/a.do","/b.do"})
/*属于前缀匹配。要以/开始。*/
//@WebFilter("/dept/*")
// 匹配所有的路径。
//@WebFilter("/*")
/*以下这个路径属于模糊匹配中的扩展匹配。以星号开始,注意这种路径不要以/开始。*/
@WebFilter("*.do")//这个是模糊匹配的写法,意思是以.do结尾的路径都可以匹配
public class Filter1 implements Filter {
public Filter1(){
System.out.println("无参数构造方法执行");
}
@Override
public void init(FilterConfig filterConfig) throws ServletException {
System.out.println("init方法执行。");
}
@Override
public void doFilter(ServletRequest request, ServletResponse
response, FilterChain chain)
throws IOException, ServletException {
// 在请求的时候添加过滤规则。
System.out.println("doFilter方法开始执行。");
//执行下一个过滤器,如果下一个不是过滤器,则执行目标程序Servlet 56
//就是向下走,没有它是不行的
chain.doFilter(request,response);
// 在响应的时候添加过滤规则。
System.out.println("doFilter方法执行结束");
}
@Override
public void destroy() {
System.out.println("destroy方法执行");
}
}
4. 注意: 56
- Servlet对象默认情况下,在服务器启动的时候是不会新建对象的。
- Filter对象默认情况下,在服务器启动的时候会新建对象。
- Servlet是单例的。Filter也是单例的。(单实例。)
4.1 目标Servlet是否执行,取决于两个条件: 56
- 第一:在过滤器当中是否编写了:chain.doFilter(request, response); 代码。
- 第二:用户发送的请求路径是否和Servlet的请求路径一致。
- chain.doFilter(request, response); 这行代码的作用:
- 执行下一个过滤器,如果下面没有过滤器了,执行最终的Servlet。
- 注意:Filter的优先级,天生的就比Servlet优先级高。
- /a.do 对应一个Filter,也对应一个Servlet。那么一定是先执行Filter,然后再执行Servlet。
4.2 关于Filter的配置路径:57
- /a.do、/b.do、/dept/save。这些配置方式都是精确匹配。
- /* 匹配所有路径。
- *.do 后缀匹配。不要以 / 开始
- /dept/* 前缀匹配。
4.3 在web.xml文件中进行配置的时候,Filter的执行顺序是什么? 57
依靠filter-mapping标签的配置位置,越靠上优先级越高。
过滤器的调用顺序,遵循栈数据结构。
代码在com.bjpowernode.javaweb.servlet
Filter2
package com.bjpowernode.javaweb.servlet;
import jakarta.servlet.*;
import java.io.IOException;
//这是一个过滤器 57
public class Filter2 implements Filter {
@Override
public void init(FilterConfig filterConfig) throws ServletException {
}
@Override
public void doFilter(ServletRequest request, ServletResponse response
, FilterChain chain) throws IOException, ServletException {
System.out.println("Filter2 doFilter begin");
chain.doFilter(request,response);
System.out.println("Filter2 doFilter end");
}
@Override
public void destroy() {
}
}
4.4 使用@WebFilter的时候,Filter的执行顺序是怎样的呢?57
- 执行顺序是:比较Filter这个类名。
- 比如:FilterA和FilterB,则先执行FilterA。
- 比如:Filter1和Filter2,则先执行Filter1.
5. Filter的生命周期?57
和Servlet对象生命周期一致。
唯一的区别:Filter默认情况下,在服务器启动阶段就实例化。Servlet不会。
6. Filter过滤器这里有一个设计模式:57
- 责任链设计模式。
6.1 过滤器最大的优点:
- 在程序编译阶段不会确定调用顺序。因为Filter的调用顺序是配置到web.xml文件中的,只要修改web.xml配置文件中filter-mapping的顺序就可以调整Filter的执行顺序。显然Filter的执行顺序是在程序运行阶段动态组合的。那么这种设计模式被称为责任链设计模式。
- 责任链设计模式最大的核心思想:
- 在程序运行阶段,动态的组合程序的调用顺序。