关于servlet:过滤器和监听器

42次阅读

共计 6902 个字符,预计需要花费 18 分钟才能阅读完成。

过滤器和监听器

次要内容

过滤器

介绍

​ Filter 即为过滤,用于在 Servlet 之外对 Request 或者 Response 进行批改。它次要用于对用户申请进行预处理,也能够对 HttpServletResponse 进行后处理。应用 Filter 的残缺流程:Filter 对用户申请进行预处理,接着将申请交给 Servlet 进行解决并生成响应,最初 Filter 再 对服务器响应进行后处理。在一个 web 利用中,能够开发编写多个 Filter,这些 Filter 组合 起来称之为一个 Filter 链。

单个过滤器

多个过滤器

​ 若是一个过滤器链:先配置先执行 (申请时的执行程序);响应时: 以相同的程序执行。

​ 在 HttpServletRequest 达到 Servlet 之前,拦挡客户的 HttpServletRequest。依据须要查看 HttpServletRequest,也能够批改 HttpServletRequest 头和数据。

​ 在 HttpServletResponse 达到客户端之前,拦挡 HttpServletResponse。依据须要查看 HttpServletResponse,也能够批改 HttpServletResponse 头和数据。

实现

​ 能够通过实现一个叫做 javax.servlet.Fileter 的接口来实现一个过滤器,其中定义了 三个办法,init(), doFilter(), destroy() 别离在相应的机会执行。前期察看生命周期。

​ Filter 的实现只须要两步:

​ Step1: 编写 java 类实现 Filter 接口,并实现其 doFilter 办法。

​ Step2: 通过 @WebFilter 注解设置它所能拦挡的资源。

@WebFilter("/*")
public class Filter01 implements Filter {
    @Override
    public void init(FilterConfig filterConfig) throws ServletException { }

    @Override
    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException { }

    @Override
    public void destroy() {}
} 

​ Filter 接口中有一个 doFilter 办法,当开发人员编写好 Filter,并配置对哪个 web 资源进行拦挡后,Web 服务器每次在调用 web 资源的 service 办法之前,都会先调用一下 filter 的 doFilter 办法。因而能够达到如下成果:

​ 调用指标资源之前,让一段代码执行。

​ 是否调用指标资源(即是否让用户拜访 web 资源)。

​ web 服务器在调用 doFilter 办法时,会传递一个 filterChain 对象进来,filterChain 对象是 filter 接口中最重要的一个对象,它提供了一个 doFilter 办法,开发人员能够依据需要决定 是否调用此办法,调用该办法,则 web 服务器就会调用 web 资源的 service 办法,即 web 资源就会被拜访,否则 web 资源不会被拜访。(实质是放行,调用 doFilter 办法后,即申请能够达到资源)

实例

申请乱码解决
/**
 * 字符乱码解决
 *     乱码状况:Tomcat8 及以上版本                            Tomcat7 及以下版本
                             
         POST 申请        乱码,须要解决                                乱码,须要解决
                     request.setCharacterEncoding("UTF-8");            
             
         GET 申请
                     不会乱码,不须要解决                            乱码,须要解决
             new String(request.getParameter("参数名").getBytes("ISO-8859-1"),"UTF-8");
             
     
         如何解决:1、解决 POST 申请
                 request.setCharacterEncoding("UTF-8");
             2、解决 GET 申请且服务器版本在 Tomcat8 以下的
                 1> 失去申请类型(GET 申请)2> 失去服务器的版本的信息
                 3> 判断是 GET 申请且 Tomcat 版本小于 8
                 4> 解决乱码
                   new String(request.getParameter("参数名").getBytes("ISO-8859-1"),"UTF-8");
        
 */
@WebFilter("/*")
public class AEncodingFilter implements Filter {public AEncodingFilter() { }

    public void destroy() {}

    public void doFilter(ServletRequest arg0, ServletResponse arg1, FilterChain chain) throws IOException, ServletException {
        // 基于 HTTP
        HttpServletRequest request = (HttpServletRequest) arg0;
        HttpServletResponse response = (HttpServletResponse) arg1;
        
        // 解决申请乱码乱码(解决 POST 申请)request.setCharacterEncoding("UTF-8");    
        
        // 解决 GET 申请且服务器版本在 Tomcat8 以下的
        String method = request.getMethod();
        // 如果是 GET 申请
        if ("GET".equalsIgnoreCase(method)) {
            // 服务器版本在 Tomcat8 以下的 Apache Tomcat/8.0.45
            String serverInfo = request.getServletContext().getServerInfo();             
            // 失去具体的版本号
            String versionStr = serverInfo.substring(serverInfo.indexOf("/")+1, serverInfo.indexOf("."));
            // 判断服务器版本是否小于 8
            if (Integer.parseInt(versionStr) < 8) {
                // 失去自定义外部类(MyWapper 继承了 HttpServletRequestWapper 对象,而 HttpServletRequestWapper 对象实现了 HttpServletRequest 接口,所以 MyWapper 的实质也是 request 对象)HttpServletRequest myRequest = new MyWapper(request);
                // 放行资源
                chain.doFilter(myRequest, response);
                return;
            }
        }
        
        // 放行资源
        chain.doFilter(request, response);        
    }

    public void init(FilterConfig fConfig) throws ServletException { }
    
    
    /**
     * 定义外部类,继承 HttpServletRequestWrapper 包装类对象,重写 getParameter() 办法
     */
    class MyWapper extends HttpServletRequestWrapper {
        
        // 定义成员变量,晋升结构器 中的 request 对象的范畴
        private HttpServletRequest request;

        public MyWapper(HttpServletRequest request) {super(request);
            this.request = request;
        }

        /**
         * 重写 getParameter() 办法
         */
        @Override
        public String getParameter(String name) {String value = request.getParameter(name);
            
            if (value != null && !"".equals(value.trim())) {
                try {
                    // 将默认 ISO-8859- 1 编码的字符转换成 UTF-8
                    value = new String(value.getBytes("ISO-8859-1"),"UTF-8");
                } catch (UnsupportedEncodingException e) {e.printStackTrace();
                }
            }
            return value;
        }
    }
} 
用户非法拜访拦挡
/**
 * 非法拜访拦挡(当用户未登录时,拦挡申请到登录页面)*         拦挡的资源:*             拦挡所有资源  /*
 *         须要被放行的资源:*             不须要登录即可拜访的资源
 *             1、放行指定页面,不须要登录能够拜访的页面(例如:登录页面、注册页面等)*             2、放行动态资源(例如:css、js、image 等资源)*             3、放行指定操作,不须要登录即可执行的操作(例如:登录操作、注册操作等)*             4、登录状态放行(如果存在指定 sessuin 对象,则为登录状态)*/    
@WebFilter("/*")
public void doFilter(ServletRequest arg0, ServletResponse arg1, FilterChain chain) throws IOException, ServletException {
        
        // 基于 HTTP
        HttpServletRequest request = (HttpServletRequest) arg0;
        HttpServletResponse response = (HttpServletResponse) arg1;
        // 失去申请的门路
        String path = request.getRequestURI(); //     站点名 / 资源门路    
        // 1、放行指定页面,不须要登录能够拜访的页面(例如:登录页面、注册页面等)if (path.contains("/login.jsp") || path.contains("/register.jsp")) {chain.doFilter(request, response);
             return;
        }
        // 2、放行动态资源(例如:css、js、image 等资源)if (path.contains("/js")) {chain.doFilter(request, response);
             return;
        }
        // 3、放行指定操作,不须要登录即可执行的操作(例如:登录操作、注册操作等)if (path.contains("/loginServlet")) {chain.doFilter(request, response);
             return;
        }
        // 4、登录状态放行(如果存在指定 sessuin 对象,则为登录状态)// 失去 session 域对象
        String uname = (String) request.getSession().getAttribute("user");
        // 如果 session 域对象不为空,则为登录状态,放行资源
        if (uname != null && !"".equals(uname.trim())) {chain.doFilter(request, response);
             return;
        }    
        
        // 若以上条件均不满足,拦挡跳转到登录页面
        response.sendRedirect("login.jsp");
        return;
    } 

监听器

介绍

​ web 监听器是 Servlet 中一种的非凡的类,能帮忙开发者监听 web 中的特定事件,比方 ServletContext,HttpSession,ServletRequest 的创立和销毁;变量的创立、销毁和批改等。能够在某些动作前后减少解决,实现监控。例如能够用来统计在线人数等。

实现

​ 监听器有三类 8 种:

​ ⑴ 监听生命周期:

​ ServletRequestListener

​ HttpSessionListener

​ ServletContextListener

​ ⑵ 监听值的变动:

​ ServletRequestAttributeListener

​ HttpSessionAttributeListener

​ ServletContextAttributeListener

​ ⑶ 针对 session 中的对象:

​ 监听 session 中的 java 对象 (javaBean),是 javaBean 间接实现监听器 的接口。

示例

​ 做一个对在线人数的监控。

 实现步骤:

​ Step1:创立一个监听器,须要实现某种接口,依据需要选取 HttpSessionListener

​ Step2:通过 @WebListener 注解配置该监听器

​ 创立一个类,并实现 HttpSessionListener 接口,用来检测 Session 的创立和销毁。

1. 在类中定义一个成员变量用来存储以后的 session 个数。(OnlineListener.java)

/**
 * 在线人数统计
 *     当有新的 session 对象被创立,则在线人数 +1;*     有 session 对象被销毁,在线人数 -1;* @author Lisa Li
 *
 */
@WebListener
public class OnlineListener implements HttpSessionListener {
    
    // 默认在线人数
    private Integer onlineNumber = 0;

    /**
     * 当有新的 session 对象被创立,则在线人数 +1;*/
    @Override
    public void sessionCreated(HttpSessionEvent se) {
        // 人数 +1
        onlineNumber++;
        // 将人数存到 session 作用域中
        // se.getSession().setAttribute("onlineNumber", onlineNumber);
        // 将人数存到 application 作用域中
        se.getSession().getServletContext().setAttribute("onlineNumber", onlineNumber);
    }

    /**
     * 有 session 对象被销毁,在线人数 -1;*/
    @Override
    public void sessionDestroyed(HttpSessionEvent se) {
        // 人数 -1
        onlineNumber--;
        // 将人数存到 session 作用域中
        // se.getSession().setAttribute("onlineNumber", onlineNumber);
        // 将人数存到 application 作用域中
        se.getSession().getServletContext().setAttribute("onlineNumber", onlineNumber);
    }

} 

2. 做一个测试的 Servlet 用来登录,和显示以后在线人数。(OnlineServlet.java)

/**
 * 在线人数统计
 */
public class OnlineServlet extends HttpServlet {
    private static final long serialVersionUID = 1L;
    
    protected void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        
        // 失去参数
        String key = request.getParameter("key");
        
        // 判断是否为空 (不为空,且值为 logout 则为退出操作)
        if (key != null && "logout".equals(key)) {
            // 传递了参数,示意要做用户退出操作
            request.getSession().invalidate();
            return;
        }
            
        // 创立 session 对象
        HttpSession session = request.getSession();
        // 获取 sessio 作用域中的在线人数
        Integer onlineNumber = (Integer) session.getServletContext().getAttribute("onlineNumber");                        

        // 输入
        response.setContentType("text/html;charset=UTF-8");
        response.getWriter().write("<h2> 在线人数:"+onlineNumber+"</h2><h4><a href='online?key=logout'> 退出 </a><h4>");    
    
    }
} 

equest.getSession();
// 获取 sessio 作用域中的在线人数
Integer onlineNumber = (Integer) session.getServletContext().getAttribute(“onlineNumber”);

 // 输入
    response.setContentType("text/html;charset=UTF-8");
    response.getWriter().write("<h2> 在线人数:"+onlineNumber+"</h2><h4><a href='online?key=logout'> 退出 </a><h4>");    

} 

}

正文完
 0