在乐字节学习的一天(在线更新)

在乐字节学习的第8天,学习了挺多知识,下面就是我今天在乐字节学习的知识。

介绍

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("/*")

publicclassFilter01implementsFilter{

@Override

publicvoidinit(FilterConfigfilterConfig)throwsServletException{

   }

@Override

publicvoiddoFilter(ServletRequestservletRequest,ServletResponseservletResponse,FilterChainfilterChain)throwsIOException,ServletException{

   }

@Override

publicvoiddestroy() {

   }

}

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("/*")

publicclassAEncodingFilterimplementsFilter{


publicAEncodingFilter() {

   }

    publicvoiddestroy() {   

    }

    publicvoiddoFilter(ServletRequestarg0,ServletResponsearg1,FilterChainchain)throwsIOException,ServletException{

        // 基于HTTP

        HttpServletRequestrequest=(HttpServletRequest)arg0;

        HttpServletResponseresponse=(HttpServletResponse)arg1;


        // 处理请求乱码乱码 (处理POST请求)

        request.setCharacterEncoding("UTF-8"); 


        // 处理GET请求且服务器版本在Tomcat8以下的

        Stringmethod=request.getMethod();

        // 如果是GET请求

        if("GET".equalsIgnoreCase(method)) {

            // 服务器版本在Tomcat8以下的 Apache Tomcat/8.0.45

            StringserverInfo=request.getServletContext().getServerInfo();         

            // 得到具体的版本号

            StringversionStr=serverInfo.substring(serverInfo.indexOf("/")+1,serverInfo.indexOf("."));

            // 判断服务器版本是否小于8

            if(Integer.parseInt(versionStr)<8) {

                // 得到自定义内部类  (MyWapper继承了HttpServletRequestWapper对象,而HttpServletRequestWapper对象实现了HttpServletRequest接口,所以MyWapper的本质也是request对象)

                HttpServletRequestmyRequest=newMyWapper(request);

                // 放行资源

                chain.doFilter(myRequest,response);

                return;

            }

        }


        // 放行资源

        chain.doFilter(request,response);     

    }

    publicvoidinit(FilterConfigfConfig)throwsServletException{


    }



    /**

    * 定义内部类,继承HttpServletRequestWrapper包装类对象,重写getParameter()方法

    */

    classMyWapperextendsHttpServletRequestWrapper{


        // 定义成员变量,提升构造器 中的request对象的范围

        privateHttpServletRequestrequest;

        publicMyWapper(HttpServletRequestrequest) {

            super(request);

            this.request=request;

        }

        /**

        * 重写getParameter()方法

        */

        @Override

        publicStringgetParameter(Stringname) {

            Stringvalue=request.getParameter(name);


            if(value!=null&&!"".equals(value.trim())) {

                try{

                    // 将默认ISO-8859-1编码的字符转换成UTF-8

                    value=newString(value.getBytes("ISO-8859-1"),"UTF-8");

                }catch(UnsupportedEncodingExceptione) {

                    e.printStackTrace();

                }

            }

            returnvalue;

        }

    }

}

用户非法访问拦截

/**

* 非法访问拦截(当用户未登录时,拦截请求到登录页面)

*    拦截的资源:

*        拦截所有资源  /*

*    需要被放行的资源:

*        不需要登录即可访问的资源

*        1、放行指定页面,不需要登录可以访问的页面 (例如:登录页面、注册页面等)

*        2、放行静态资源(例如:css、js、image等资源)

*        3、放行指定操作,不需要登录即可执行的操作(例如:登录操作、注册操作等)

*        4、登录状态放行 (如果存在指定sessuin对象,则为登录状态)

*/

@WebFilter("/*")

publicvoiddoFilter(ServletRequestarg0,ServletResponsearg1,FilterChainchain)throwsIOException,ServletException{


        // 基于HTTP

        HttpServletRequestrequest=(HttpServletRequest)arg0;

        HttpServletResponseresponse=(HttpServletResponse)arg1;

        // 得到请求的路径

        Stringpath=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域对象

        Stringuname=(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

publicclassOnlineListenerimplementsHttpSessionListener{


    // 默认在线人数

    privateIntegeronlineNumber=0;

    /**

    * 当有新的session对象被创建,则在线人数+1;

    */

    @Override

    publicvoidsessionCreated(HttpSessionEventse) {

        // 人数+1

        onlineNumber++;

        // 将人数存到session作用域中

        // se.getSession().setAttribute("onlineNumber", onlineNumber);

        // 将人数存到application作用域中

        se.getSession().getServletContext().setAttribute("onlineNumber",onlineNumber);

    }

    /**

    * 有session对象被销毁,在线人数-1;

    */

    @Override

    publicvoidsessionDestroyed(HttpSessionEventse) {

        // 人数-1

        onlineNumber--;

        // 将人数存到session作用域中

        // se.getSession().setAttribute("onlineNumber", onlineNumber);

        // 将人数存到application作用域中

        se.getSession().getServletContext().setAttribute("onlineNumber",onlineNumber);

    }

}

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

/**

* 在线人数统计

*/

publicclassOnlineServletextendsHttpServlet{

    privatestaticfinallongserialVersionUID=1L;


    protectedvoidservice(HttpServletRequestrequest,HttpServletResponseresponse)throwsServletException,IOException{


        // 得到参数

        Stringkey=request.getParameter("key");


        // 判断是否为空 (不为空,且值为logout则为退出操作)

        if(key!=null&&"logout".equals(key)) {

// 传递了参数,表示要做用户退出操作

            request.getSession().invalidate();

return;

       }


// 创建session对象

HttpSessionsession=request.getSession();

// 获取sessio作用域中的在线人数

IntegeronlineNumber=(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>");


    }

}

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,186评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,858评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,620评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,888评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,009评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,149评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,204评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,956评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,385评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,698评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,863评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,544评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,185评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,899评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,141评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,684评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,750评论 2 351