过滤器的四种拦截方式

四种拦截方式

引言

我们先做一个小测试,新建一个b.jsp:

<body>
	<h1>b.jsp</h1>
</body>

再新建一个a.jsp,在里面转发到b.jsp:

<body>
	<h1>a.jsp</h1>
	<%
		request.getRequestDispatcher("/b.jsp").forward(request, response);
	%>
</body>

然后我们配置一个过滤器,指定过滤的资源为b.jsp:

public class MyFilter implements Filter {
	public void doFilter(ServletRequest request, ServletResponse response,
			FilterChain chain) throws IOException, ServletException {
		System.out.println("MyFilter...");
		chain.doFilter(request, response);
	}
}

xml:

<filter>
	<display-name>MyFilter</display-name>
	<filter-name>MyFilter</filter-name>
	<filter-class>com.veeja.web.filter.MyFilter</filter-class>
</filter>
<filter-mapping>
	<filter-name>MyFilter</filter-name>
	<url-pattern>/b.jsp</url-pattern>
</filter-mapping>

然后我们在浏览器中直接访问b.jsp,显然的是,过滤器是会执行的:
在这里插入图片描述
但是,如果我们访问a.jsp呢,我们是知道的,a.jsp会转发到b.jsp,这样过滤器会执行吗?我们试一试:
在这里插入图片描述
结果我们发现,这个时候过滤器却没有执行。这是为什么呢?

这是因为,在默认情况下,只能直接访问目标资源才会执行过滤器,而forward执行目标资源,就不会执行过滤器。

四种拦截

其实过滤器有四种拦截方式!分别是:REQUESTFORWARDINCLUDEERROR

  • REQUEST:直接访问目标资源时执行过滤器。包括:在地址栏中直接访问、表单提交、超链接、重定向,只要在地址栏中可以看到目标资源的路径,就是REQUEST;
  • FORWARD:转发访问执行过滤器。包括RequestDispatcher#forward()方法、<jsp:forward>标签都是转发访问;
  • INCLUDE:包含访问执行过滤器。包括RequestDispatcher#include()方法、<jsp:include>标签都是包含访问;
  • ERROR:当目标资源在web.xml中配置为<error-page>中时,并且真的出现了异常,转发到目标资源时,会执行过滤器。

可以在<filter-mapping>中添加0~n个<dispatcher>子元素,来说明当前访问的拦截方式。

接下来我们举几个例子,说明一下应用的实际情况:

<filter-mapping>
	<filter-name>myfilter</filter-name>
	<url-pattern>/b.jsp</url-pattern>
	<dispatcher>REQUEST</dispatcher>
	<dispatcher>FORWARD</dispatcher>
</filter-mapping>

这个配置中,b.jsp为目标资源。
当直接请求b.jsp时,会执行过滤器。
当转发到b.jsp页面时,会执行过滤器。

<filter-mapping>
	<filter-name>myfilter</filter-name>
	<url-pattern>/b.jsp</url-pattern>
</filter-mapping>

这里没有给出拦截方式时,那么默认为REQUEST。这也能解释我们在引言中的例子了。

<filter-mapping>
	<filter-name>myfilter</filter-name>
	<url-pattern>/b.jsp</url-pattern>
	<dispatcher>FORWARD</dispatcher>
</filter-mapping>

这一个配置中,当转发到b.jsp页面时,会执行过滤器。而且因为已经给出了<dispatcher>FORWARD</dispatcher>了,那么就没有默认的REQUEST了。所以只有在转发到b.jsp时才会执行过滤,而直接访问到b.jsp时,不会执行过滤器。

接下来,我们再举一个ERROR拦截的例子:

<filter-mapping>
	<filter-name>myfilter</filter-name>
	<url-pattern>/b.jsp</url-pattern>
	<dispatcher>ERROR</dispatcher>
</filter-mapping>
<error-page>
	<error-code>500</error-code>
	<location>/b.jsp</location>
</error-page>

上面里例子中,拦截的方式为ERROR,而且把b.jsp设置为HTTPS 500的错误页面。

我们写一个jsp,为a.jsp:

<body>
	<h1>a.jsp</h1>
	<%
		if(true)
			throw new RuntimeException("hahhaahahaha~");
	%>
</body>

这个时候,当用户访问a.jsp页面时会抛出异常,也就是500错误。
这时服务器会转发到b.jsp,而在这之前会执行上面配置的过滤器。

其实最为常用的就是REQUEST和FORWARD两种拦截方式,而INCLUDE和ERROR都比较少用。大家多做练习,就能熟练掌握了。


以上就是全部内容,谢谢你的阅读。

如有纰漏,请不吝指出,不胜感激。

如果你不沉下心来认真练习,所有的阅读都是白费的。

  • 3
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值