大家好,这里是教授.F
引入:
为什么需要过滤器???我们在访问一个项目的时候,常常有很多页面,如果没有过滤器,则我们需要在用户访问一个页面的时候,都要进行一个校验,这样就非常代码就很冗余。
工作机制:
细节:
1.Filter的配置跟servlet非常相似,因为Filter也是被tomcat管理和维护的
2.
在 web.xml 配置过滤器
<!--filter 标签用于配置一个 Filter 过滤器-->
<filter>
<!-- filter 别名-->
<filter-name>ManageFilter</filter-name>
<!-- filter 类全路径-->
<filter-class>com.hspedu.servlet.ManageFilter</filter-class>
</filter>
<filter-mapping>
<!-- filter 别名 -->
<filter-name>ManageFilter</filter-name>
<!-- 老韩解读(重要)
1. url-pattern 配置拦截路径 /manage/*
2. 第 1 个 / 被服务器解析为:http://ip:port/工程路径/
3. /manage/* 表示:http://ip:port/工程路径/manage/* 所有资源请求都经过该过滤器
-->
<url-pattern>/manage/*</url-pattern>
</filter-mapping>
上面的第三点说到的所有的资源请求都经过该过滤器。这个怎么理解???
如果我们登录一个页面的时候,还没进行登录验证就显示出该内容,这样显然不行,所以我们把内容放在一个指定的地方(例如上面说的,放在manage中),这样只有在登录后,在能显示出manage的内容。看下面的图片:
我们使用路径定位来操作。内容只放在manage中。这样就能保证:如果没有登录,是不能看到mangae的内容的。
3.在每次调用Filter时,Filter中的doFilter()就会被调用
4.在销毁Filter时,Filter中的destroy()就会被调用
5.每次启动tomcat的时候,Filter会自动被调用
url-pattern:
生命周期:
1.filter在web项目启动时,有tomcat来创建filter实例,只会创建一个
2.会调用filter默认的无参构造器,同时会调用init方法,只会调用一次
3.在创建filter实例时,同时会创建一个filterConfig对象,并通过init方法传入
4.通过FilterConfig对象,程序员可以获取该filter的相关配置信息
5.当一个http请求和该filter的url-patter匹配时,就会调用deFilter方法
6.在调用doFilter方法时,tomcat会同时创建servletRequest 和 servletResponse
和 Filter Chain对象,并通过doFilter传入
7.如果后面的请求目标资源(jsp,servlet)会 使用到request 和 response ,那么会继续传递
FilterConfig():
1. FilterConfig 是 Filter 过滤器的配置类
2. Tomcat 每次创建 Filter 的时候,也会创建一个 FilterConfig 对象,这里包含了Filter 配置文件的配置信息。
3. FilterConfig 对象作用是获取 filter 过滤器的配置内容
过滤器链:
FilterChain: 在处理某些复杂业务时,一个过滤器不够,可以设计多个过滤器共同完成过滤任务,形成过滤器链。
注意:过滤器链的执行顺序跟在web.xml配置的顺序一样
FilterChain 注意事项和细节:
1. 多个 filter 和目标资源在一次 http 请求,在同一个线程中
2. 当一个请求 url 和 filter 的 url-pattern 匹配时, 才会被执行, 如果有多个匹配上,就会顺序执行,形成一个 filter 调用链(底层可以使用一个数据结构搞定)
3. 多个 filter 共同执行时,因为是一次 http 请求, 使用同一个request 对象
4. 多个 filter 执行顺序,和 web.xml 配置顺序保持一致.
5. chain.doFilter(req, resp)方法 将执行下一个过滤器的 doFilter 方法, 如果后面没有过滤器,则执行目标资源。
6. 小结:注意执行过滤器链时, 顺序是(用前面的案例分析) Http请求-> A 过滤器dofilter()-> A 过滤器前置代码 -> A 过滤器 chain.doFilter() -> B 过滤器dofilter() -> B 过滤器前置代码 -> B过滤器 chain.doFilter() -> 目标文件 -> B过滤器后置代码-> A过滤器后置代码->返回给浏览器页面/数据