热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

浅谈监听器与过滤器

ListenerServlet的应用Listener是Servlet的监听器,它可以监听客户端的请求、服务端的操作等。通过监听器,可以自动激发一些操作&
Listener Servlet的应用
Listener是 Servlet的监听器,它可以监听客户端的请求、服务端的操作等。通过监听器,可以自动激发一些操作,比如监听在线的用户的数量。当增加一个HttpSession时,就激发sessionCreated(HttpSessionEvent se)方法,这样就可以给在线人数加1。常用的监听接口有以下几个:
    
 ServletContextAttributeListener监听对 ServletContext属性的操作,比如增加、删除、修改属性。
    
 ServletContextListener监听 ServletContext。当创建ServletContext时,激发contextInitialized(ServletContextEvent sce)方法;当销毁ServletContext时,激发contextDestroyed(ServletContextEvent sce)方法。
    
 HttpSessionListener监听 HttpSession的操作。当创建一个Session时,激发session Created(HttpSessionEvent se)方法;当销毁一个Session时,激发sessionDestroyed (HttpSessionEvent se)方法。
    
 HttpSessionAttributeListener监听 HttpSession中的属性的操作。当在Session增加一个属性时,激发attributeAdded(HttpSessionBindingEvent se) 方法;当在Session删除一个属性时,激发attributeRemoved(HttpSessionBindingEvent se)方法;当在Session属性被重新设置时,激发attributeReplaced(HttpSessionBindingEvent se) 方法。 

    下面我们开发一个具体的例子,这个监听器能够统计在线的人数。在ServletContext初始化和销毁时,在服务器控制台打印对应的信息。当ServletContext里的属性增加、改变、删除时,在服务器控制台打印对应的信息。 

    要获得以上的功能,监听器必须实现以下3个接口:
    
 HttpSessionListener 
    
 ServletContextListener 
    
 ServletContextAttributeListener 

    我们看具体的代码,见示例 14-9。 


    【程序源代码】 


1 // ==================== Program Discription =====================
2 // 程序名称:示例 14-9 : EncodingFilter .java
3 // 程序目的:学习使用监听器
4 // ==============================================================
5 import javax.servlet.http.*;
6 import javax.servlet.*;
7
8 public class OnLineCountListener implements HttpSessionListener,
ServletContextListener,ServletContextAttributeListener
9 {
10 private int count;
11 private ServletContext context = null;
12 
13 public OnLineCountListener()
14 {
15 count=0;
16 //setContext();
17 }
18 //创建一个 session时激发
19 public void sessionCreated(HttpSessionEvent se)
20 {
21 count++;
22 setContext(se);
23 
24 }
25 //当一个 session失效时激发
26 public void sessionDestroyed(HttpSessionEvent se)
27 {
28 count--;
29 setContext(se);
30 }
31 //设置 context的属性,它将激发attributeReplaced或attributeAdded方法
32 public void setContext(HttpSessionEvent se)
33 {
34 se.getSession().getServletContext().
setAttribute("onLine",new Integer(count));
35 }
36 //增加一个新的属性时激发
37 public void attributeAdded(ServletContextAttributeEvent event) {
38 
39 log("attributeAdded('" + event.getName() + "', '" +
40 event.getValue() + "')");
41 
42 }
43 
44 //删除一个新的属性时激发
45 public void attributeRemoved(ServletContextAttributeEvent event) {
46
47 log("attributeRemoved('" + event.getName() + "', '" +
48 event.getValue() + "')");
49 
50 }
51
52 //属性被替代时激发
53 public void attributeReplaced(ServletContextAttributeEvent event) {
54 
55 log("attributeReplaced('" + event.getName() + "', '" +
56 event.getValue() + "')");
57 }
58 //context删除时激发
59 public void contextDestroyed(ServletContextEvent event) {
60 
61 log("contextDestroyed()");
62 this.context = null;
63 
64 }
65 
66 //context初始化时激发
67 public void contextInitialized(ServletContextEvent event) {
68 
69 this.context = event.getServletContext();
70 log("contextInitialized()");
71 
72 }
73 private void log(String message) {
74 
75 System.out.println("ContextListener: " + message);
76 }  
77 }


    【程序注解】 
    在OnLineCountListener里,用count代表当前在线的人数,OnLineCountListener将在Web服务器启动时自动执行。当OnLineCountListener构造好后,把count设置为0。每增加一个Session,OnLineCountListener会自动调用sessionCreated(HttpSessionEvent se)方法;每销毁一个Session,OnLineCountListener会自动调用sessionDestroyed(HttpSessionEvent se)方法。当调用sessionCreated(HttpSessionEvent se)方法时,说明又有一个客户在请求,此时使在线的人数(count)加1,并且把count写到ServletContext中。ServletContext的信息是所有客户端共享的,这样,每个客户端都可以读取到当前在线的人数。 

为了使监听器生效,需要在web.xml里进行配置,如下所示: 


OnLineCountListener


测试程序: 


<%&#64; page contentType&#61;"text/html;charset&#61;gb2312" %>


目前在线人数&#xff1a; 


<%&#61;getServletContext().getAttribute("onLine")%>


退出会话&#xff1a; 




getServletContext().getAttribute("onLine")获得了 count的具体值。客户端调用 


<%session.invalidate() ;%>


    使 Session失效&#xff0c;这样监听器就会使count减1。 

    【运行程序】
    web.xml做好以上的配置&#xff0c;把OnLineCountListener放在WEB-INF/class目录下&#xff0c;启动Web服务器&#xff0c;在浏览器里输入以下URL&#xff08;根据具体情况不同&#xff09;&#xff1a;http://127.0.0.1:8080/ch14/listener.jsp 

    浏览器将会打印目前在线人数。在服务器端有以下输出&#xff1a; 


ContextListener: contextInitialized()
ContextListener: attributeReplaced(&#39;org.apache.
catalina.WELCOME_FILES&#39;, &#39;[Ljava.lang.String;&#64;1d98a&#39;)
ContextListener: attributeAdded(&#39;onLine&#39;, &#39;1&#39;)
ContextListener: attributeReplaced(&#39;onLine&#39;, &#39;1&#39;)
ContextListener: attributeReplaced(&#39;onLine&#39;, &#39;0&#39;)
ContextListener: attributeReplaced(&#39;onLine&#39;, &#39;1&#39;)
ContextListener: attributeReplaced(&#39;onLine&#39;, &#39;2&#39;)
Servlet开发和配置过滤器
JSP技术构建在 Servlet技术之上&#xff0c;所以Servlet和JSP的技术本质是一样的&#xff0c;JSP能做到的&#xff0c;Servlet都能做到&#xff0c;但是它们却各有所长。Servlet比较适合作为控制类组件&#xff0c;比如视图控制器等。另外&#xff0c;Servlet还可以作为过滤器、监听器等。Servlet不仅可以动态生成HTML内容&#xff0c;还可以动态生成图形。总而言之&#xff0c;Servlet在项目中作为控制类的组件&#xff0c;并且处理一些后台业务&#xff0c;JSP则作为显示组件。 

    在本节&#xff0c;我们将介绍Servlet常用的使用方法之一&#xff1a;作为过滤器。在Servlet作为过滤器使用时&#xff0c;它可以对客户的请求进行过滤处理&#xff0c;当它处理完成后&#xff0c;它会交给下一个过滤器处理&#xff0c;就这样&#xff0c;客户的请求在过滤链里一个个处理&#xff0c;直到请求发送到目标。举个例子&#xff0c;某个网站里有提交"修改的注册信息"的网页&#xff0c;当用户填写完成修改信息并提交后&#xff0c;服务端在进行真正的处理时需要做两个处理&#xff1a;客户端的会话是否有效&#xff1b;对提交的数据进行统一的编码&#xff0c;比如GB2312。这两个处理可以在由两个过滤器组成的过滤链里进行处理。当过滤器处理成功后&#xff0c;把提交的数据发送到最终目标&#xff1b;如果过滤器处理不成功&#xff08;比如客户端的会话无效&#xff09;&#xff0c;它将把视图派发到指定的错误页面。可以看出&#xff0c;过滤器就像一扇门&#xff0c;客户端要和服务端的某个目标交互&#xff0c;必须通过这扇门。 

    下面我们来看一个具体的例子&#xff0c;这个例子将介绍怎么开发过滤器&#xff0c;并且介绍怎么在web.xml文件里配置过滤器。这个例子里有两个JSP页面&#xff0c;前一个页面用户输入一些信息然后提交&#xff0c;后一个页面显示用户提交的信息。在提交信息后&#xff0c;要经过两个过滤器的处理&#xff0c;一个检查用户是否登录&#xff0c;一个把用户的提交信息用GB2312进行重新编码。 

    开发一个Filter&#xff0c;这个Filter需要实现Filter接口&#xff0c;Filter接口定义了以下的方法&#xff1a; 


destroy() //由 Web容器调用&#xff0c;销毁此Filter
init(FilterConfig filterConfig) ///由 Web容器调用&#xff0c;初始化此Filter
doFilter(ServletRequest request, ServletResponse response,
FilterChain chain)//具体过滤处理代码


    下面我们来看对提交信息用 GB2312进行重新编码的Filter&#xff0c;见示例14-7、示例14-8。

    【程序源代码】 


1 // &#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61; Program Discription &#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;
2 // 程序名称&#xff1a;示例 14-7 : EncodingFilter .java
3 // 程序目的&#xff1a;学习使用编码过滤器
4 // &#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;
5 import javax.servlet.FilterChain;
6 import javax.servlet.ServletRequest;
7 import javax.servlet.ServletResponse;
8 import java.io.IOException;
9 import javax.servlet.Filter;
10 import javax.servlet.http.HttpServletRequest;
11 import javax.servlet.http.HttpServletResponse;
12 import javax.servlet.ServletException;
13 import javax.servlet.FilterConfig;
14 
15 public class EncodingFilter implements Filter
16 {
17 
18 private String targetEncoding &#61; "gb2312";
19 protected FilterConfig filterConfig;
20 
21 public void init(FilterConfig config) throws ServletException {
22 this.filterConfig &#61; config;
23 this.targetEncoding &#61; config.getInitParameter("encoding");
24 }
25 
26 
27 public void doFilter(ServletRequest srequest,
ServletResponse sresponse,FilterChain chain)
28 throws IOException, ServletException {
29 
30 HttpServletRequest request &#61; (HttpServletRequest)srequest;
31 request.setCharacterEncoding(targetEncoding);//把请求用指定的方式编码
32 // 把处理发送到下一个过滤器
33 chain.doFilter(srequest,sresponse); 
34 }
35 
36 public void destroy()
37 {
38 this.filterConfig&#61;null;
39 }
40 
41 public void setFilterConfig(final FilterConfig filterConfig)
42 {
43 this.filterConfig&#61;filterConfig;
44 }
45 }


    【程序源代码】 


1 // &#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61; Program Discription &#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;
2 // 程序名称&#xff1a;示例 14-8 : LoginFilter.java
3 // 程序目的&#xff1a;学习使用登录过滤器
4 // &#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;
5 import javax.servlet.FilterChain;
6 import javax.servlet.ServletRequest;
7 import javax.servlet.ServletResponse;
8 import java.io.IOException;
9 import javax.servlet.Filter;
10 import javax.servlet.http.HttpServletRequest;
11 import javax.servlet.http.HttpServletResponse;
12 import javax.servlet.ServletException;
13 import javax.servlet.FilterConfig;
14
15 public class LoginFilter implements Filter
16 {
17 String LOGIN_PAGE&#61;"init.jsp";
18 protected FilterConfig filterConfig;
19 public void doFilter(final ServletRequest req,final ServletResponse
res,FilterChain chain)throws IOException,ServletException
20 {
21 HttpServletRequest hreq &#61; (HttpServletRequest)req;
22 HttpServletResponse hres &#61; (HttpServletResponse)res;        
23 String isLog&#61;(String)hreq.getSession().getAttribute("isLog");        
24 if((isLog!&#61;null)&&((isLog.equals("true"))||(isLog&#61;&#61;"true")))//检查是否登录
25 {
26 chain.doFilter(req,res);
27 return ;
28 }
29 else
30 hres.sendRedirect(LOGIN_PAGE);//如果没有登录&#xff0c;把视图派发到登录页面
31 }
32 
33 public void destroy()
34 {
35 this.filterConfig&#61;null;
36 }
37 public void init(FilterConfig config)
38 {
39 this.filterConfig&#61;config;
40 }
41 public void setFilterConfig(final FilterConfig filterConfig)
42 {
43 this.filterConfig&#61;filterConfig;
44 }       
45 }


    【程序注解】 
    正如前面所说&#xff0c;EncodingFilter的目的是把客户端的请求用指定的方式编码&#xff0c;具体的处理在request.setCharacterEncoding(targetEncoding)完成了。LoginFilter判断用户在进入目标之前是否登录&#xff0c;if((isLog!&#61;null)&&((isLog.equals("true"))||(isLog&#61;&#61;"true")))将检查用户是否登录&#xff0c;如果已登录&#xff0c;那么把视图让过滤链继续处理&#xff0c;如果没有登录&#xff0c;把视图派发到登录页面&#xff0c;过滤链处理结束。
下面我们来看怎么在web.xml里配置这两个过滤器&#xff0c;代码如下所示&#xff1a;
    【程序源代码】 


encoding
EncodingFilter
encoding
gb2312
   
auth
LoginFilter
encoding
/* 
 
auth
/target.jsp
        


    【程序注解】 
    可以看出&#xff0c;配置Filter时&#xff0c;首先指定Filter的名字和Filter的实现类&#xff0c;如果有必要&#xff0c;还要配置Filter的初始参数&#xff1b;然后为Filter做映射&#xff0c;这个映射指定了需要过滤的目标&#xff08;JSP、Servlet&#xff09;。在上面的例子中&#xff0c;指定了EncodingFilter 为所有的JSP和Servlet做过滤&#xff0c;LoginFilter为target.jsp做过滤。这样&#xff0c;当客户请求target.jsp时&#xff0c;首先要经过EncodingFilter的处理&#xff0c;然后经过LoginFilter的处理&#xff0c;最后才把请求传递给target.jsp。 

    【运行程序】
    把程序部署到Web服务器里&#xff08;比如Tomcat&#xff09;&#xff0c;然后启动Web服务器&#xff0c;在浏览器里输入以下URL&#xff08;根据具体请求改变URL&#xff09;&#xff1a;http://127.0.0.1:8080/ch14/target.jsp 

    那么Filter将会把视图派发到&#xff1a;http://127.0.0.1:8080/ch14/init.jsp 

    在init.jsp里&#xff0c;我们使用&#xff1a; 


<% session.setAttribute("isLog","true");%>


    来设置用户已经登录&#xff08;这里是简化的&#xff0c;在实际项目中&#xff0c;可能要经过验证处理&#xff09;。在 init.jsp里&#xff0c;可以提交一些中文的信息。由于提交的信息被EncodingFilter使用GB2312统一编码了&#xff0c;故在target.jsp里能够正确显示中文。您可以做一个试验&#xff0c;把 


encoding
/* 


改为 


encoding
/nothing 


    然后重新启动 Web服务器。那么在target.jsp里&#xff0c;中文将不能正确显示。

原文链接&#xff1a;http://www.dlog.cn/nicholascoder/diary/9282

转载于:https://www.cnblogs.com/chenying99/archive/2012/09/04/2670344.html


推荐阅读
  • CSS3选择器的使用方法详解,提高Web开发效率和精准度
    本文详细介绍了CSS3新增的选择器方法,包括属性选择器的使用。通过CSS3选择器,可以提高Web开发的效率和精准度,使得查找元素更加方便和快捷。同时,本文还对属性选择器的各种用法进行了详细解释,并给出了相应的代码示例。通过学习本文,读者可以更好地掌握CSS3选择器的使用方法,提升自己的Web开发能力。 ... [详细]
  • 在Android开发中,使用Picasso库可以实现对网络图片的等比例缩放。本文介绍了使用Picasso库进行图片缩放的方法,并提供了具体的代码实现。通过获取图片的宽高,计算目标宽度和高度,并创建新图实现等比例缩放。 ... [详细]
  • android listview OnItemClickListener失效原因
    最近在做listview时发现OnItemClickListener失效的问题,经过查找发现是因为button的原因。不仅listitem中存在button会影响OnItemClickListener事件的失效,还会导致单击后listview每个item的背景改变,使得item中的所有有关焦点的事件都失效。本文给出了一个范例来说明这种情况,并提供了解决方法。 ... [详细]
  • Imtryingtofigureoutawaytogeneratetorrentfilesfromabucket,usingtheAWSSDKforGo.我正 ... [详细]
  • 本文介绍了如何使用PHP向系统日历中添加事件的方法,通过使用PHP技术可以实现自动添加事件的功能,从而实现全局通知系统和迅速记录工具的自动化。同时还提到了系统exchange自带的日历具有同步感的特点,以及使用web技术实现自动添加事件的优势。 ... [详细]
  • Skywalking系列博客1安装单机版 Skywalking的快速安装方法
    本文介绍了如何快速安装单机版的Skywalking,包括下载、环境需求和端口检查等步骤。同时提供了百度盘下载地址和查询端口是否被占用的命令。 ... [详细]
  • 本文介绍了在开发Android新闻App时,搭建本地服务器的步骤。通过使用XAMPP软件,可以一键式搭建起开发环境,包括Apache、MySQL、PHP、PERL。在本地服务器上新建数据库和表,并设置相应的属性。最后,给出了创建new表的SQL语句。这个教程适合初学者参考。 ... [详细]
  • 这是原文链接:sendingformdata许多情况下,我们使用表单发送数据到服务器。服务器处理数据并返回响应给用户。这看起来很简单,但是 ... [详细]
  • 向QTextEdit拖放文件的方法及实现步骤
    本文介绍了在使用QTextEdit时如何实现拖放文件的功能,包括相关的方法和实现步骤。通过重写dragEnterEvent和dropEvent函数,并结合QMimeData和QUrl等类,可以轻松实现向QTextEdit拖放文件的功能。详细的代码实现和说明可以参考本文提供的示例代码。 ... [详细]
  • 本文分享了一个关于在C#中使用异步代码的问题,作者在控制台中运行时代码正常工作,但在Windows窗体中却无法正常工作。作者尝试搜索局域网上的主机,但在窗体中计数器没有减少。文章提供了相关的代码和解决思路。 ... [详细]
  • Linux重启网络命令实例及关机和重启示例教程
    本文介绍了Linux系统中重启网络命令的实例,以及使用不同方式关机和重启系统的示例教程。包括使用图形界面和控制台访问系统的方法,以及使用shutdown命令进行系统关机和重启的句法和用法。 ... [详细]
  • 本文介绍了Redis的基础数据结构string的应用场景,并以面试的形式进行问答讲解,帮助读者更好地理解和应用Redis。同时,描述了一位面试者的心理状态和面试官的行为。 ... [详细]
  • 如何使用Java获取服务器硬件信息和磁盘负载率
    本文介绍了使用Java编程语言获取服务器硬件信息和磁盘负载率的方法。首先在远程服务器上搭建一个支持服务端语言的HTTP服务,并获取服务器的磁盘信息,并将结果输出。然后在本地使用JS编写一个AJAX脚本,远程请求服务端的程序,得到结果并展示给用户。其中还介绍了如何提取硬盘序列号的方法。 ... [详细]
  • t-io 2.0.0发布-法网天眼第一版的回顾和更新说明
    本文回顾了t-io 1.x版本的工程结构和性能数据,并介绍了t-io在码云上的成绩和用户反馈。同时,还提到了@openSeLi同学发布的t-io 30W长连接并发压力测试报告。最后,详细介绍了t-io 2.0.0版本的更新内容,包括更简洁的使用方式和内置的httpsession功能。 ... [详细]
  • 本文介绍了Web学习历程记录中关于Tomcat的基本概念和配置。首先解释了Web静态Web资源和动态Web资源的概念,以及C/S架构和B/S架构的区别。然后介绍了常见的Web服务器,包括Weblogic、WebSphere和Tomcat。接着详细讲解了Tomcat的虚拟主机、web应用和虚拟路径映射的概念和配置过程。最后简要介绍了http协议的作用。本文内容详实,适合初学者了解Tomcat的基础知识。 ... [详细]
author-avatar
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有