过滤器&监听器

Filter:过滤器

概念

  • 当访问服务器的资源时,过滤器可以将请求拦截下来,完成一些特殊的功能。
  • 作用:一般用于完成通用的操作。如:登录验证、统一编码处理、敏感字符过滤

快速入门

  • 定义一个类,实现Filter接口(javax.servlet.filter)
  • 复写方法
  • 配置拦截路径(拦截准备去访问的资源)
    • web.xml
    • 注解(@WebFilter)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
@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("Filter过滤器被执行了");
// 放行
filterChain.doFilter(servletRequest, servletResponse);
}
@Override
public void destroy() {
}
}

过滤器细节

  • web.xml配置

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    <filter>
    <filter-name>demo1</filter-name>
    <filter-class>com.nogizaka.filter.FilterDemo1</filter-class>
    </filter>

    <filter-mapping>
    <filter-name>demo1</filter-name>
    <!--拦截路径-->
    <url-pattern>/*</url-pattern>
    </filter-mapping>
  • 过滤器执行流程

    1. 执行过滤器
    2. 执行放行后的资源
    3. 回来执行过滤器放行代码下的代码
  • 过滤器生命周期方法

    1. init:在服务器启动后,会创建Filter对象,然后调用init方法。只执行一次。用于加载资源

    2. doFilter:每一次请求被拦截资源时,会执行。执行多次

    3. destroy:在服务器关闭后,Filter对象被销毁。如果服务器是正常关闭,则会执行destroy方法。只执行一次。用于释放资源

  • 过滤器配置详解

    • 拦截路径配置:

      1. 具体资源路径:/index.jsp,只有访问index.jsp资源时,过滤器才会被执行
      2. 拦截目录:/user/*,访问/user下的所有资源时,过滤器都会被执行
      3. 后缀名拦截:*.jsp,访问所有后缀名为jsp资源时,过滤器都会被执行
      4. 拦截所有资源:/*,访问所有任意资源时,过滤器都会被执行
    • 拦截方式配置:

      • 注解配置:设置dispatcherTypes的属性

        • REQUEST:默认值,浏览器直接请求资源
        • FORWARD:转发访问资源
        • INCLUDE:包含访问资源
        • ERROR:错误跳转资源
        • ASYNC:异步访问资源
      • web.xml

        • 设置<dispatcher>></dispatcher>标签
  • 过滤器链(配置多个过滤器)

    • 执行顺序:如果有两个过滤器1,过滤器2

      • 过滤器1->过滤器2->资源执行->过滤器2->过滤器1
    • 过滤器先后顺序:

      • 注解配置:按照类名的字符串比较规则,值小的先执行
      • web.xml配置:<filter-mapper>谁先定义谁先执行

代理模式

  • 真实对象:被代理的对象
  • 代理对象
  • 代理模式:代理对象 代理 真实对象,达到增强真实对象功能的目的
  • 实现方法:

    • 静态代理:有一个类文件描述代理模式
    • 动态代理:在内存中形成代理类

      • 实现步骤:

        1. 代理对象和真实对象实现相同的接口
        2. 代理对象 = Proxy.newProxyInstance()
        3. 使用代理对象调用方法
        4. 增强方法
      • 增强方式

        1. 增强参数列表
        2. 增强返回值类型
        3. 增强方法体执行逻辑
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
43
44
45
46
public class ProxyDemo {
public static void main(String[] args) {
// 1.创建真实对象
Apple apple = new Apple();

// 2. 动态代理增强Apple对象

/*
* 三个对象:
* 1.类加载器:真实对象.getClass().getClassLoader()
* 2.接口数组:真实对象.getClass().getInterfaces()
* 3.处理器:使用匿名内部类 new InvocationHandler()
* */
SaleComputer proxy_apple = (SaleComputer) Proxy.newProxyInstance(apple.getClass().getClassLoader(), apple.getClass().getInterfaces(), new InvocationHandler(){
/*
* 代理逻辑编写的方法:代理对象调用的所有方法都会触发该方法执行
* 参数:
* 1. proxy:代理对象
* 2. method:代理对象的调用的方法,被封装为的对象
* 3. args:代理对象调用的方法时,传递的实际参数
* */
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
// 判断是否是sale方法
if(method.getName()=="sale"){
// 增强参数
double money = (double) args[0];
money += 1000;
// 使用真实对象调用该方法
String obj = (String) method.invoke(apple, money);
return obj+"和鼠标";
}else {
// 不是sale方法,将实际参数返回
Object obj = method.invoke(apple, args);
return obj;
}

}
});

//3.调用方法
String computer = proxy_apple.sale(8000);
System.out.println(computer);

}
}

Listener:监听器

  • 概念:web的三大组件之一

    • 事件监听机制
      • 事件 :一件事情
      • 事件源 :事件发生的地方
      • 监听器 :一个对象
      • 注册监听:将事件、事件源、监听器绑定在一起。 当事件源上发生某个事件后,执行监听器代码
  • ServletContextListener:监听ServletContext对象的创建和销毁

    • 方法

      • void contextDestroyed(ServletContextEvent sce) :ServletContext对象被销毁之前会调用该方法
      • void contextInitialized(ServletContextEvent sce) :ServletContext对象创建后会调用该方法
    • 步骤:

      1. 定义一个类,实现ServletContextListener接口
      2. 复写方法
      3. 配置

        • 注解:@WebListener

        • web.xml

1
2
3
4
5
6
7
<listener>
<listener-class>全类名</listener-class>
</listener>

<!--指定初始化参数-->
<context-param>
</context-param>
Knowledge is priceless, thanks for your support !