javaweb三大组件之过滤器Filter
一、过滤器执行流程
1.执行过滤器
2.执行放行后的资源
3.回来执行过滤器放行代码下边的代码
二、过滤器生命周期方法
1.init:在服务器启动后,会创建Filter对象,然后调用init方法。只执行一次。用于加载资源
2.doFilter:每一次请求被拦截资源时,会执行。执行多次
3.destroy:在服务器关闭后,Filter对象被销毁。如果服务器是正常关闭,则会执行destroy方法。只执行一次。用于释放资源
三、过滤器配置详解
1.拦截路径配置:
- 具体资源路径: /index.jsp,只有访问index.jsp资源时,过滤器才会被执行
- 拦截目录: /user/*,访问/user下的所有资源时,过滤器都会被执行
- 后缀名拦截: *.jsp,访问所有后缀名为jsp资源时,过滤器都会被执行
- 拦截所有资源:/*,访问所有资源时,过滤器都会被执行
2.拦截方式配置:资源被访问的方式
注解配置:设置dispatcherTypes属性
- REQUEST:默认值。浏览器直接请求资源
- FORWARD:转发访问资源
- INCLUDE:包含访问资源
- ERROR:错误跳转资源
- ASYNC:异步访问资源
如:
@WebFilter(value="/*",dispatcherTypes ={DispatcherType.FORWARD,DispatcherType.REQUEST})web.xml配置:
- 在< filter-mapping>标签内设置< dispatcher></ dispatcher>标签即可,标签值同注解配置方式一样
四、过滤器链(配置多个过滤器)
1.过滤器优先级:
- web.xml配置: < filter-mapping>定义在上边的优先级高于定义在下面的
- 注解配置:按照类名的字符串比较规则比较,值小的先执行。如:AFilter 和 BFilter,AFilter就先执行了。
2.执行顺序:
假设有两个过滤器,过滤器1优先于过滤器2,则执行方式为:
(1)过滤器1
(2)过滤器2
(3)资源执行
(4)过滤器2
(5)过滤器1
五、示例代码(xml方式)
FilterDemo1.java:
package cn.itcast.web.filter;
import javax.servlet.*;
import java.io.IOException;
public class FilterDemo1 implements Filter {
@Override
public void init(FilterConfig filterConfig) throws ServletException {
System.out.println("初始化方法");
}
@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() {
System.out.println("销毁方法");
}
}
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>cn.itcast.web.filter.FilterDemo1</filter-class>
</filter>
<!--配置过滤器映射规则-->
<filter-mapping>
<!--要配置的过滤器的名字-->
<filter-name>demo1</filter-name>
<!--过滤器的拦截路径-->
<url-pattern>/*</url-pattern>
</filter-mapping>
</web-app>
六、示例代码(注解方式)
FilterDemo2.java:
package cn.itcast.web.filter;
import javax.servlet.*;
import javax.servlet.annotation.WebFilter;
import java.io.IOException;
@WebFilter("/*")
public class FilterDemo2 implements Filter {
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 {
System.out.println("FilterDemo2过滤器初始化");
}
public void destroy() {
System.out.println("FilterDemo2过滤器销毁");
}
}
七、登录过滤器案例
package cn.itcast.web.filter;
import javax.servlet.*;
import javax.servlet.annotation.WebFilter;
import javax.servlet.http.HttpServletRequest;
import java.io.IOException;
/**
* 登录验证的过滤器
*/
@WebFilter("/*")
public class LoginFilter implements Filter {
public void doFilter(ServletRequest req, ServletResponse resp, FilterChain chain) throws ServletException, IOException {
System.out.println(req);
//0.强制转换
HttpServletRequest request = (HttpServletRequest) req;
//1.获取资源请求路径
String uri = request.getRequestURI();
//2.判断是否包含登录相关资源路径,要注意排除掉 css/js/图片/验证码等资源
if(uri.contains("/login.jsp") || uri.contains("/loginServlet") || uri.contains("/css/") || uri.contains("/js/") || uri.contains("/fonts/") || uri.contains("/checkCodeServlet") ){
//包含,用户就是想登录。放行
chain.doFilter(req, resp);
}else{
//不包含,需要验证用户是否登录
//3.从获取session中获取user
Object user = request.getSession().getAttribute("user");
if(user != null){
//登录了。放行
chain.doFilter(req, resp);
}else{
//没有登录。跳转登录页面
request.setAttribute("login_msg","您尚未登录,请登录");
request.getRequestDispatcher("/login.jsp").forward(request,resp);
}
}
// chain.doFilter(req, resp);
}
public void init(FilterConfig config) throws ServletException {
}
public void destroy() {
}
}