Java 杂谈善倾的知识体系构建之路

过滤器 Filter

2018-09-01  本文已影响0人  善倾

Servlet 可以分为简单 Servlet 、过滤器和监听器,过滤器本质上也是 Servlet ,只不过它是 Tomcat 管理的一种特殊的 Servlet 。

struts 框架中又把过滤器叫做拦截器,它就像 web 应用程序的一个单向进出的门。浏览器发送过来的请求由 Tomcat 解析后,Tomcat 会先将请求给过滤器,由过滤器进行过滤之后,再转给相应的 Servlet 进行处理,待此次请求响应送到浏览器去之前,最后要经过的组件也是这个过滤器。简单的说就是,在 web 应用程序前加上了一个只能够单向进出的门,request 请求只能从这里进,最后的 response 响应也只能从这里出。而且这个门的组件化程度非常的高,直接将过滤器类添加到项目中,再进行相关配置就可以使用了,完全不需要对前台页面和后台组件做相关的更改。因为 JSP 本质上也是 Servlet 类,所以过滤器对于访问 JSP 页面的请求同样有效。

要想实现一个过滤器类要遵循以下几个语法点:

过滤器链

一个 web 应用程序可以配置多个过滤器类,如果他们之间过滤的是同一个资源的话,就会以链的形式连接在一起,执行顺序就是由配置文件中配置的先后顺序确定的。Tomcat 将请求给第一个过滤器,然后第一个过滤器做了相关工作后再把请求给它的下一过滤器类,依次这样下去之后,最后把请求转发到最开始需要访问的 web 组件去。加上了过滤器的概念后,web 应用程序的执行流程图如下。

_Filter.jpg

Filter 生命周期

Filter 本质上就是 Servlet 的变形,这一点同 JSP 是一样的。所以 Filter 生命周期和 Servlet 的声明周期是一样的吗?废话哦,都是人,分工不同而已,生老病死不都是一样的,如下图所示。

_FilterLifeCycle.jpg

javax.servlet.Filter 接口的全部内容就是只有三个抽象方法,分别对应着 Filter 对象的初始化、运行和销毁状态节点。

关于doFilter()方法还有以下需要记录的

_FilterChain.jpg

Filter 在 web.xml 中的配置

Filter 本质上也是 Servlet ,所以它的配置和 Servlet 几乎是一模一样的。唯独特殊的一点是,Tomcat 启动的时候,Filter 对象就会立即被创建,如果它不早早的被创建,那怎么实现过滤器的功能呢?所以它不需要像 Servlet 类一样使用<load-on-startup>元素来指定它的加载优先级,期望它能够在 Tomcat 一启动就立即被创建。

配置方式如下:

<filter>
    <filter-name>SimpleFilter</filter-name>
    <filter-class>filter.SimpleFilter</filter-class>
</filter>
<filter-mapping>
    <filter-name>SimpleFilter</filter-name>
    <url-pattern>/*</url-pattern>
</filter-mapping>

其中<url-pattern>中的/表示当前 web 应用的根目录。如果后面接的是通配符*表示所有的请求都需要经过这个过滤器过滤,如果是*.jsp,表示过滤所有的 .jsp 文件。如果像/*.jsp这样既有路径匹配又有扩展名匹配, Tomcat 在解析 web.xml 文件的时候不知道到底要按照哪种方式匹配,就会产生 IllegalArgumentException 异常。添加初始化参数的方式和 Servlet 是一样的,都是使用的<init-param>元素包含 name 和 value 信息。

过滤器的应用

过滤器可以用来做编码过滤、登录验证以及非法文字拦截还可以做结果拦截器,还有记录系统访问日志以及利用它来记录当前页面历史访问次数,这不是博客网站都会有的一个基本功能吗?

编码过滤对拿到的数据进行指定编码设置就可以了,这样可以避免每个 Servlet 都显示的去指定编码设置,避免写大量重复代码。非法文字拦截只要对拿到的数据内容进行判断就可以了。两者的实现都很简单!

登录验证是任何一个动态网站都必备的功能了,具体是通过判断此次会话的 session 中是否有登录验证信息来确定登录状态的。这里面有一个隐式的陷阱存在,如果是第一次登录访问,那么此时服务器是没有对应的 session 的,如果这个时候还没有进行登录判断就提前被拦截器拦截,判断不处于登录状态后,就提前结束此次请求后,那么用户将永远无法登录系统。解决方法是判断请求 url ,如果是登录页面,则不进行 session 登录状态判断。

上一篇下一篇

猜你喜欢

热点阅读