Java Web 开发时,可以使用 Listener 来监听来监听一些事件,从而实现一些功能。实际上这个监听器,原理就是 Delphi 中大家常用的各种事件。

1. 那么,监听器的主要用途用哪些呢:

  • 统计在线人数和在线用户
  • 系统启动时加载初始化信息: 包括各种缓存、共公的定制器、数据库链接等等
  • 统计网站访问量
  • 路Spring结合

2. 监听器可以按监听的对象来分类:

  • ServletContext (ServletContextListener):用于监听应用程序环境对象的事件监听器(一个项目中只有一个),可以用来启动定时器、初始化全局对象。
  • HttpSession (HttpSessionListener): 用于监听用户会话对象的事件监听器。
  • ServletRequest (ServletRequestListener): 用于监听请求消息对象的事件监听器,可以用来读取参数,记录访问历史等。

3. 按监听的事件来划分:

  • 监听域对象自身的创建和销毁的事件监听器(2中的三类)。
  • 监听域对象中的属性的增加和删除的事件监听器:  ServletContextAttributeListenerHttpSessionAttributeListenerServletRequestAttributeListener
  • 监听绑定到 HttpSession 域中的某个对象的状态的事件监听器。它分为这几种状态:
    • 绑定: 将状态通过 set 保存到 Session 中。
    • 解除绑定 : 使用 remove 删除状态。
    • 钝化: 将 session 对象持久化到存储设备上。(Session本身是存在于服务器内存中。Session 钝化机制由SessionManager管理: 第一种管理器是 org.apache.catalina.session.StandardManger, 当 tomcat服务器被关闭或重启时,tomcat服务器会将当前内存中的session钝化到服务器文件系统中。另一种情况是web应用程序被重新加载时,内存中的session对象也会被钝化到服务器的文件系统中。钝化后的文件被保存到 Tomcat安装路径下的 /work/Catalina/hostname/applicationname/SESSION.ser 中。第二种管理器是 org.apache.catalina.session.Persistentmanager ,它是在钝化基础上进行了扩张,前两种情况和StandardManager相同,第三种情况,可以配置主流内存的Session对象数目,将不常用的Session对象保存到文件系统或数据库中,当要使用时再进行加载。默认情况下,Tomcat提供两个钝化驱动类:org.apache.Catalina.FileStore 和 org.apache.Catalina.JDBCStore。)
    • 活化: 从存储设备上恢复。

  绑定、解除绑定使用 HttpSessionBindingListener 接口, 钝化、活化使用 HttpSessionActivationListener 接口。这两个监听器不需要注册。

4. 注册方法

在 Servlet 3.0 之前的版本中, 需要在 web.xml 中进行注册。也比较简单,就是一个声明:

<listener>
<listener-class>com.imooc.listener.FirstListener</listener-class>
</listener>

在 servlet 3.0 之后, 可以不再到 web.xml 中进行配置, 直接给监听器类加上注解 @WebListener 就可以。

package com.imooc.listener;

import javax.servlet.ServletContextEvent;
import javax.servlet.ServletContextListener;
import javax.servlet.annotation.WebListener; import com.demo.utils.UserManager; @WebListener
public class FirstListener implements ServletContextListener { @Override
public void contextInitialized(ServletContextEvent sce) {
System.out.println("contextInitialized");
} @Override
public void contextDestroyed(ServletContextEvent sce) {
System.out.println("contextDestroyed");
} }

5. 示例: 使用监听器实现的显示在线用户列表

效果如下:

此示例综合用到了这些技术: jstl和el标签、jsp脚本、listener监听器 、单例全局对象、javabean、线程同步。

主要代码:

index.jsp

<%@ page language="java" contentType="text/html; charset=UTF-8" %>
<%@ page import="com.demo.utils.*" %>
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %>
<%@ taglib uri="http://java.sun.com/jsp/jstl/functions" prefix="fn" %>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>欢迎访问</title>
</head>
<body>
<% request.setAttribute("users", UserManager.getInstance().getItems()); %>
<h1>在线用户列表</h1><hr>
<!-- 使用 JSTL 输出在线用户列表 -->
<table>
<tr><th width="80px">name</th><th width="320px">sessionID</th><th width="180px">IP地址</th><th>登录时间</th><tr>
<c:forEach var="user" items="${requestScope.users}">
<tr>
<td>${user.name}</td>
<td>${user.sessionID}</td>
<td>${user.ipaddr}</td>
<td>${user.firstTimeStr}</td>
</tr>
</c:forEach></table>
<br>
当前在线: <c:out value="${fn:length(users)}"></c:out>人。
</body>
</html>

RequestListener.java

package com.demo.listener;

import javax.servlet.ServletRequestEvent;
import javax.servlet.ServletRequestListener;
import javax.servlet.annotation.WebListener;
import javax.servlet.http.HttpServletRequest; import com.demo.utils.User;
import com.demo.utils.UserManager; @WebListener
public class RequestListener implements ServletRequestListener { @Override
public void requestDestroyed (ServletRequestEvent sre) {
System.out.println("requestDestroyed");
} @Override
public void requestInitialized (ServletRequestEvent sre) {
HttpServletRequest req = (HttpServletRequest) sre.getServletRequest();
UserManager mgr = UserManager.getInstance();
// 用户请求时,如果 mgr 中不存在当前会放原 sessionID, 则新建一个User对象,加入管理器中
if (!mgr.existSession(req.getSession().getId())) {
User item = new User();
item.setFirstTime(System.currentTimeMillis());
item.setIpaddr(req.getRemoteAddr() + ":" + req.getRemotePort());
item.setName(req.getParameter("name"));
item.setSessionID(req.getSession().getId());
mgr.addSession(item);
}
System.out.println("requestInitialized: " + req.getRequestURI() + ", " + req.getParameter("name"));
}
}

SessionListener.java

package com.demo.listener;

import javax.servlet.annotation.WebListener;
import javax.servlet.http.HttpSessionEvent;
import javax.servlet.http.HttpSessionListener; import com.demo.utils.UserManager; @WebListener
public class SessionListener implements HttpSessionListener { @Override
public void sessionCreated(HttpSessionEvent se) {
System.out.println("sessionCreated");
} @Override
public void sessionDestroyed(HttpSessionEvent se) {
System.out.println("sessionDestroyed");
// Session 失效时,从列表中删除
UserManager.getInstance().removeSession(se.getSession().getId());
}
}

UserManager.java

package com.demo.utils;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map; /** 用户管理 */
public class UserManager {
private static UserManager Instance; private Map<String, User> items = new HashMap<String, User>(); /** 单例 */
public static UserManager getInstance() {
if (Instance == null)
Instance = new UserManager();
return Instance;
} private UserManager() {} // 由于 Web 请求是并发的,对列表的操作,需要使用 synchronized 关键字线程同步,防止出现异常
public synchronized boolean existSession(String sessionID) {
return items.containsKey(sessionID);
} public synchronized void addSession(User v) {
if (v != null)
items.put(v.getSessionID(), v);
} public synchronized void removeSession(String sessionID) {
if (items.containsKey(sessionID))
items.remove(sessionID);
} public int size() {
return items.size();
} public synchronized User get(String sessionID) {
return items.get(sessionID);
} public synchronized List<User> getItems() {
ArrayList<User> list = new ArrayList<User>();
for (Map.Entry<String, User> entry : items.entrySet())
list.add(entry.getValue());
return list;
}
}

User.java

package com.demo.utils;

import java.io.Serializable;
import java.text.SimpleDateFormat; /** 用户对象 Java Bean */
public class User implements Serializable {
private static final long serialVersionUID = 1L; private String name;
private String sessionID;
private String ipaddr;
private long firstTime; public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getSessionID() {
return sessionID;
}
public void setSessionID(String sessionID) {
this.sessionID = sessionID;
}
public String getIpaddr() {
return ipaddr;
}
public void setIpaddr(String ipaddr) {
this.ipaddr = ipaddr;
}
public long getFirstTime() {
return firstTime;
}
public void setFirstTime(long firstTime) {
this.firstTime = firstTime;
} public String getFirstTimeStr() {
return new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(firstTime).toString();
}
}

完整项目下载:

链接: http://pan.baidu.com/s/1jI2fVqm 密码: wjr5

【感谢】

慕课网、Fcming 老师

[Java] JSP笔记 - Listener 监听器的更多相关文章

  1. java新手笔记27 监听器类

    1.外部类监听 package com.yfs.javase; import java.awt.Button; import java.awt.FlowLayout; import java.awt. ...

  2. Java中的Listener 监听器

    Listener的定义与作用 监听器Listener就是在application,session,request三个对象创建.销毁或者往其中添加修改删除属性时自动执行代码的功能组件. Listener ...

  3. JAVA JSP笔记

    一.jsp加载项目中资源图片 如果直接将静态页面写的代码copy到jsp中,你会发现图片都无法加载. 获取代码: String path = request.getContextPath(); Str ...

  4. [Java] JSP笔记 - 自定义标签

    自定义标签的创建步骤: 自定义标签的四大功能: 自定义标签的类结构: 在 1.0 中呢, 可以将 <body-content> 的值设置为 JSP, 2.0中则不允许在自定义标签体中出现j ...

  5. [Java] JSP笔记 - EL、JSTL 常用标签

    一. 什么是 EL 语言 表达式语言(EL)是 JSP 2.0 引入的一种计算和输出 Java 对象的简单语言. 二.EL 语言的作用 为了使JSP写起来更加简单.表达式语言的灵感来自于 ECMASc ...

  6. [Java] JSP笔记 - Filter 过滤器

    一.什么是Web过滤器 Servlet API 很久以前就已成为企业应用开发的基石,而 Servlet 过滤器则是对 J2EE 家族的相对较新的补充. Servlet 过滤器是可插入的 Web 组件, ...

  7. [Java] JSP笔记 - Java Bean

    一. Java Bean 是什么? Java Bean 其实就是一个符合特定规则的 Java Class.这些规则包括: 使用 public 声明的公用类 属性使用public 的get,set方法访 ...

  8. 【Servlet】Java Serlvet Listener 监听器

    Listener监听器 Servlet规范中定义的一种特殊的组件,用来监听Servlet容器产生的事件并进行相应的处理 容器产生的事件分类 - 生命周期相关的事件 - 设置和删除Attribute数据 ...

  9. .jsp文件的使用和理解以及一些小练习和Listener监听器

    什么是 jsp,它有什么用? jsp 的全换是 java server pages.Java 的服务器页面.jsp 的主要作用是代替 Servlet 程序回传 html 页面的数据.因为 Servle ...

随机推荐

  1. 【译】Spring 4 Hello World例子

    前言 译文链接:http://websystique.com/spring/spring-4-hello-world-example-annotation-tutorial-full-example/ ...

  2. Oracle数据库,数据的增、删、改、查

    oracle数据库中,数据的增.删.改.查,通过SQL语句实现 SQL:结构化查询语言: 特点:不区分大小写:字符串用单引号引起来:语句结束用分号表示结束: 行注释,在语句的最前面加"--& ...

  3. Windows Server 2012 虚拟化实战:SCVMM的安装和部署

    本篇大概介绍一下在Windows Server 2012 R2上安装和部署SCVMM的过程及其注意事项.下图是我们数据中心SCVMM的基本架构,其中 SCVMM Database 是用于存储了所有配置 ...

  4. 64位下pwntools中dynELF函数的使用

    这几天有同学问我在64位下怎么用这个函数,于是针对同一道题写了个利用dynELF的方法 编译好的程序 http://pan.baidu.com/s/1jImF95O 源码在后面 from pwn im ...

  5. java Future 接口介绍

    (转自:http://blog.csdn.net/yangyan19870319/article/details/6093481) 在Java中,如果需要设定代码执行的最长时间,即超时,可以用Java ...

  6. MMORPG大型游戏设计与开发(服务器 AI 事件)

    AI中的事件与场景中的事件大致相同,都是由特定的条件触发的.只不过AI的事件与其他事件不同的是,对于AI的事件往往是根据不同的AI类型,和动态的触发条件下才产生的.其实不管AI多么智能,它对应的触发条 ...

  7. [Top-Down Approach] Chatper 3 Notes

    这里留下空白,提醒自己,第一章第二章尚待整理回顾. 此处缺了3.6/3.7两节拥塞控制的内容

  8. java基础学习03(java基础程序设计)

    java基础程序设计 一.完成的目标 1. 掌握java中的数据类型划分 2. 8种基本数据类型的使用及数据类型转换 3. 位运算.运算符.表达式 4. 判断.循环语句的使用 5. break和con ...

  9. Js函数function基础理解

    正文:我们知道,在js中,函数实际上是一个对象,每个函数都是Function类型的实例,并且都与其他引用类型一样具有属性和方法.因此,函数名实际上是指向函数对象的指针,不与某个函数绑定.在常见的两种定 ...

  10. 安全测试 - XSS如何防御

    XSS主要是通过劫持用户COOKIE,执行JS脚本进行攻击 如何发现: 可以使用<script>alert(/yourname/)</script> script最具有代表性也 ...