Filter&Listener
Filter:过滤器
概述
-
在生活中的过滤器:净水器,空气净化器,土匪
-
web中的过滤器:当访问服务器的资源时,过滤器可以将请求拦截下来,完成一些特殊的功能
-
过滤器的作用
- 一般用于完成通用的操作。如:登录操作、统一编码处理、敏感字符过滤…
快速入门
步骤
- 定义一个类,实现接口Filter
- 复写方法
- 配置拦截路径
- web.xml
- 注解
代码实现
import javax.servlet.*;
import javax.servlet.annotation.WebFilter;
import java.io.IOException;
/**
* @author zhuxianglong
* @version 1.0
* @date 2021/8/30 22:25
* 过滤器快速入门
*/
@WebFilter("/*")//访问所有资源之前,都会执行该过滤器
public class FilterDemo1 implements Filter {
@Override
public void init(FilterConfig filterConfig) throws ServletException {
}
@Override
public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
System.out.println("filterdemo1被执行了...");
//放行
filterChain.doFilter(servletRequest,servletResponse);
}
@Override
public void destroy() {
}
}
过滤器细节
web.xml配置
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"
version="4.0">
<filter>
<filter-name>demo1</filter-name>
<filter-class>net.seehope.web.filter.FilterDemo1</filter-class>
</filter>
<filter-mapping>
<filter-name>demo1</filter-name>
<%--拦截路径--%>
<url-pattern>/*</url-pattern>
</filter-mapping>
</web-app>
过滤器执行流程
- 执行过滤器
- 执行放行后的资源
- 回来执行过滤器放行代码下边的代码
import javax.servlet.*;
import javax.servlet.annotation.WebFilter;
import java.io.IOException;
/**
* @author zhuxianglong
* @version 1.0
* @date 2021/8/30 22:41
*/
@WebFilter("/*")
public class FilterDemo2 implements Filter {
public void destroy() {
}
public void doFilter(ServletRequest req, ServletResponse resp, FilterChain chain) throws ServletException, IOException {
//对request对象的请求消息增强
System.out.println("filterDemo2执行了");
//放行
chain.doFilter(req, resp);
//对response对象的响应消息增强
System.out.println("filterDemo2回来了");
}
public void init(FilterConfig config) throws ServletException {
}
}
过滤器生命周期方法
- init:在服务器启动后,会创建Filter对象调用init方法。执行一次。用于加载资源
- doFilter:每一次请求被拦截资源时,会执行。执行多次
- destroy:在服务器关闭后,Filter对象被销毁,如果服务器是正常关闭,则会执行destroy方法。执行一次。用于释放资源
过滤器配置详解
-
拦截路径配置:
- 具体资源路径:/index.jsp 只有访问index.jsp资源时,过滤器才会被执行
- 拦截目录:/user/* 访问/user下的所有资源时,过滤器都会被执行
- 后缀名拦截:*.jsp 访问所有后缀名为jsp资源时,过滤器都会被执行
- 拦截所有资源:/* 访问所有资源时,过滤器都会被执行
-
拦截方式配置:资源被访问的方式
-
注解配置
-
设置
dispatcheTypes
属性:- REQUEST:默认值。浏览器直接请求支援
- FORWARD:转发访问资源
FilterDemo4.java
import javax.servlet.*; import javax.servlet.annotation.WebFilter; import java.io.IOException; /** * @author zhuxianglong * @version 1.0 * @date 2021/8/30 22:56 */ //浏览器直接请求index.jsp资源时,该过滤器会被执行 //@WebFilter(value="/index.jsp",dispatcherTypes = DispatcherType.REQUEST) //只有转发访问index.jsp时,该过滤器才会被执行 //@WebFilter(value="/index.jsp",dispatcherTypes = DispatcherType.FORWARD) //浏览器直接请求index.jsp资源时或者转发访问index.jsp时,该过滤器才会被执行 @WebFilter(value="/index.jsp",dispatcherTypes = {DispatcherType.FORWARD,DispatcherType.REQUEST}) public class FilterDemo4 implements Filter { public void doFilter(ServletRequest req, ServletResponse resp, FilterChain chain) throws ServletException, IOException { System.out.println("filterDemo4..."); chain.doFilter(req, resp); } public void init(FilterConfig config) throws ServletException { } public void destroy() { } }
ServletDemo1.java
import javax.servlet.ServletException; import javax.servlet.annotation.WebServlet; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException; /** * @author zhuxianglong * @version 1.0 * @date 2021/8/30 23:02 */ @WebServlet("/servletDemo1") public class ServletDemo1 extends HttpServlet { protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { //转发到index.jsp request.getRequestDispatcher("/index.jsp").forward(request,response); } protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { this.doPost(request, response); } }
-
INCLUDE:包含访问资源
-
ERROR:错误跳转资源
-
ASYNC:异步访问资源
-
-
web.xml配置
- 设置
<dispatcher></dispatcher>
标签即可
- 设置
-
过滤器链(配置多个过滤器)
- 执行顺序:如果有两个过滤器:过滤器1和过滤器2
- 过滤器1
- 过滤器2
- 资源执行
- 过滤器2
- 过滤器1
- 过滤器先后顺序问题:
- 注解配置:按照类名的字符串比较规则比较,值小的先执行
- 如:AFilter和BFilter,AFilter先执行了
- web.xml:
<filter-mapping>
谁定义在上边,谁先执行
- 注解配置:按照类名的字符串比较规则比较,值小的先执行
案例
案例一:登录验证
需求
- 访问case案例的资源。验证其是否登录
- 如果登陆了,则直接放行
- 如果没有登录,则跳转到登录页面,提示“您尚未登录,请先登录”
分析
案例二:敏感词汇过滤
需求
- 对case案例录入的数据进行敏感词汇过滤
- 如果是敏感词汇,替换为***
Listener监听器
概述
- 概念:web的三大组件之一
- 挤时间监听机制
- 事件:一件事情
- 事件源:事件发生的地方
- 监听器:一个对象
- 注册监听:将事件、事件源、监听器绑定在一起。当事件源上发生某个事件后,执行监听器代码
- 挤时间监听机制
ServletContextListener
- 监听ServletContext对象的创建和销毁
方法
void contextDestriyed(ServletContextEven sce)
:ServletContext对象被销毁之前会调用该方法void contextInitialized(ServletContextEven sce)
:ServletContext对象创建后会调用该方法
步骤
-
定义一个类,实现ServletContextListener接口
-
复写方法
-
配置
-
web.xml
<listener> <listener-class>net.seehope.web.listener.ContextLoaderListener</listener-class> </listener>
- 指定初始化参数
<context-param>
- 指定初始化参数
-
注解
@WebListener
-
import javax.servlet.ServletContext;
import javax.servlet.ServletContextEvent;
import javax.servlet.ServletContextListener;
import javax.servlet.annotation.WebListener;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
/**
* @author zhuxianglong
* @version 1.0
* @date 2021/8/30 23:50
*/
@WebListener
public class ContextLoaderListener implements ServletContextListener {
/**
* 监听ServletContext对象创建的。ServletContext对象服务器启动后启动创建
* @param sce
*/
@Override
public void contextInitialized(ServletContextEvent sce) {
//加载资源文件
//1.获取ServletContext对象
ServletContext servletContext = sce.getServletContext();
//2.加载资源文件
String initParameter = servletContext.getInitParameter("contextConfigLocation");
//3.获取真实路径
String realPath = servletContext.getRealPath("contextConfigLocation");
//4.加载进内存
try {
FileInputStream fis = new FileInputStream(realPath);
System.out.println(fis);
} catch (FileNotFoundException e) {
e.printStackTrace();
}
System.out.println("ServletContext对象被创建了...");
}
/**
* 在服务器关闭后,ServletContext对象被销毁。当服务器正常关闭后该方法调用
* @param sce
*/
@Override
public void contextDestroyed(ServletContextEvent sce) {
System.out.println("ServletContext对象被销毁了...");
}
}