精华内容
下载资源
问答
  • JAVA监听
    万次阅读 多人点赞
    2018-08-15 00:45:49

    作为JAVA开发人员,我们应对javaweb的三个技术要点之一的Listener进行学习,楼主查阅相关资料并学习相关视频,总结如下JAVA监听器内容,供大家参考学习。完整代码内容我已放到我的github中,请自行下载:https://github.com/Mr-WangZhe/Java-Listener“>JAVA Listener学习

    一.监听器Listener

    1.什么是监听器

    • 监听器就是监听某个域对象的的状态变化的组件
    • 监听器的相关概念:
      • 事件源:被监听的对象(三个域对象 request、session、servletContext)
      • 监听器:监听事件源对象事件源对象的状态的变化都会触发监听器(6+2)
      • 注册监听器:将监听器与事件源进行绑定
      • 响应行为:监听器监听到事件源的状态变化时所涉及的功能代码(程序员编写代码)

    2.监听器有哪些

    • 第一维度按照被监听的对象划分:ServletRequest域、HttpSession域、ServletContext域
    • 第二维度按照监听的内容分:监听域对象的创建与销毁的、监听域对象的属性变化的

    这里写图片描述

    3.监听三大域对象的创建与销毁的监听器

    • 监听器的编写步骤(重点):

      • 编写一个监听器类去实现监听器接口
      • 覆盖监听器的方法
      • 需要在web.xml中进行配置—注册
    • 监听ServletContext域的创建与销毁的监听器ServletContextListener

      • Servlet域的生命周期

        • 何时创建:服务器启动创建
        • 何时销毁:服务器关闭销毁
      • ServletContextListener监听器的主要作用

        • 初始化的工作:初始化对象、初始化数据(加载数据库驱动、连接池的初始化)
        • 加载一些初始化的配置文件(spring的配置文件)
        • 任务调度(定时器—Timer/TimerTask)
      • 实例应用:

      • 整体流程:create包下创建MyServletContextListener类实现ServletContextListener接口并覆盖方法public void contextInitialized(ServletContextEvent arg0)监听context域对象的创建;覆盖方法public void contextDestroyed(ServletContextEvent arg0)监听context域对象的销毁。并在web.xml中通过<listener>中的<listener-class>书写监听器全类名配置。

      • 实现任务调度(即定时器)的思路:使用Timer类对象的scheduleAtFixedRate(task,firstTime,period)方法第一个参数书写TimerTask()的匿名内部类重写run方法从而实现调度任务内容;firstTime是Date类型对象,我们可以设置成当前晚上12点;period是间隔执行时间(单位毫秒),我们可以设置成24小时。从而实现从当前晚上12点开始,每24小时都执行对应的调度任务。

      • 实例代码:

        //web.xml
        <?xml version="1.0" encoding="UTF-8"?>
        <web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://java.sun.com/xml/ns/javaee" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" id="WebApp_ID" version="2.5">
        <!-- 注册监听器 -->
        <listener>
            <listener-class>create.MyServletContextListener</listener-class>
        </listener>
        <display-name>WEB23_LISTENER</display-name>
        <welcome-file-list>
          <welcome-file>index.jsp</welcome-file>
        </welcome-file-list>
        </web-app>
        
        //MyServletContextListener.java
        package create;
        import java.text.ParseException;
        import java.text.SimpleDateFormat;
        import java.util.Date;
        import java.util.Timer;
        import java.util.TimerTask;
        import javax.servlet.ServletContext;
        import javax.servlet.ServletContextEvent;
        import javax.servlet.ServletContextListener;
        public class MyServletContextListener implements ServletContextListener{
        //监听context域对象的创建
        @Override
        public void contextInitialized(ServletContextEvent arg0) {//参数可以获得被监听的对象
        //        ServletContext servletContext1 = arg0.getServletContext();//返回值是被监听的对象
        //        Object servletContext2 = arg0.getSource();//等同于arg0.getServletContext(),获得的是Object类型,但实际上也是获得被监听的对象
        //        System.out.println("context创建了...");
            //开启一个计息任务调度——每天晚上12点计息一次
            Timer timer = new Timer();
            //task任务,firstTime第一次执行的时间,period间隔执行的时间(单位毫秒)
            //timer.scheduleAtFixedRate(task, firstTime, period);
            //测试demo:从服务器启动开始每隔5秒打印“银行计息了”
        //        timer.scheduleAtFixedRate(new TimerTask() {
        //            @Override
        //            public void run() {
        //                System.out.println("银行计息了...");
        //            }
        //        }, new Date(), 5000);
            //实际银行计息业务
            //1.起始时间,定义成晚上12点
            //2.间隔时间:24小时
            SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd hh:mm:ss");
            String currentTime = "2018-08-15 00:00:00";
            Date date = null;
            try {
                date = format.parse(currentTime);
            } catch (ParseException e) {
                e.printStackTrace();
            }
            timer.scheduleAtFixedRate(new TimerTask() {
                @Override
                public void run() {
                    System.out.println("银行计息了...");
                }
            }, date, 24*60*60*1000);
        }
        //监听context域对象的销毁
        @Override
        public void contextDestroyed(ServletContextEvent arg0) {
            System.out.println("context销毁了...");
        }
        }
    • 监听Httpsession域的创建于销毁的监听器HttpSessionListener

      • HttpSession对象的生命周期

        • 何时创建:第一次调用request.getSession时创建
        • 何时销毁:服务器关闭销毁、session过期(默认30分钟,修改默认的30分钟是在Tomcat的web.xml,修改当前项目的过期时间是在自己项目的web.xml中)、手动销毁
      • HttpSessionListener监听器的主要作用:

        • 由于每次访问网站都会默认创建session对象(jsp页面中page指令中的session属性默认为true,即被访问时创建session),可以用于计数网站访问过的人
      • 实例应用:

        • 整体流程:创建MyHttpSessionListener类实现HttpSessionListener接口并覆盖public void sessionCreated(HttpSessionEvent arg0)public void sessionDestroyed(HttpSessionEvent arg0)方法。并在web.xml中注册该listener,和index.jsp页面。每当访问一次index页面,即会调用一次sessionCreated方法。

        • 实例代码:

        //web.xml
        <?xml version="1.0" encoding="UTF-8"?>
        <web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://java.sun.com/xml/ns/javaee" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" id="WebApp_ID" version="2.5">
        <!-- 注册监听器 -->
        <listener>
            <listener-class>create.MyHttpSessionListener</listener-class>
        </listener>
        <display-name>WEB23_LISTENER</display-name>
        <welcome-file-list>
          <welcome-file>index.jsp</welcome-file>
        </welcome-file-list>
        </web-app>
        
        //MyHttpSessionListener.java
        package create;
        import javax.servlet.http.HttpSessionEvent;
        import javax.servlet.http.HttpSessionListener;
        public class MyHttpSessionListener implements HttpSessionListener{
        @Override
        public void sessionCreated(HttpSessionEvent arg0) {
            //获得Session对象的方法arg0.getSession()
            String id = arg0.getSession().getId();
            System.out.println("session创建"+id);
        
        }
        @Override
        public void sessionDestroyed(HttpSessionEvent arg0) {
            System.out.println("session销毁");
        
        }
        }
        
        //index.jsp
        <%@ page language="java" contentType="text/html; charset=UTF-8"
          pageEncoding="UTF-8"%>
        <!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>Insert title here</title>
        </head>
        <body>
        index.jsp
        </body>
        </html>
    • 监听ServletRequest域创建与销毁的监听器ServletRequestListener

      • ServletRequest的生命周期
        • 创建:每一次请求都会创建request
        • 销毁:请求结束
      • 用法同上,用处不是很大,此处省略。

    4.监听三大域对象的属性变化的

    • 域对象的通用的方法:

      • setAttribute(name,value)
        • 触发添加属性的监听器的方法
        • 触发修改属性的监听器的方法
      • getAttribute(name)
      • removeAttribute(name)
        • 触发删除属性的监听器的方法
    • ServletContextAttibuteListener监听器

      • 整体使用流程:定义类MyServeltContextAttributeListener实现ServeltContextAttributeListener接口并覆盖public void attributeAdded(ServletContextAttributeEvent arg0) ;public void attributeRemoved(ServletContextAttributeEvent arg0);public void attributeReplaced(ServletContextAttributeEvent arg0)三个方法。在attributeAdded方法中的参数传递的ServletContextAttributeEvent实例对象中,调用getName方法获取放到域中的name,调用getValue方法获取放到域中的value;在attributeRemoved方法中的参数传递的ServletContextAttributeEvent实例对象中,调用getName方法获取被删除的域中的name,调用getValue方法获取被删除的域中的value;在attributeReplaced 方法中的参数传递的ServletContextAttributeEvent实例对象中,调用getName方法获取修改前的域中的name,调用getValue方法获取修改前的域中的value。并在web.xml中注册监听器。

      • 实例代码:

      //MyServletContextAttributeLister.java
      package attribute;
      import javax.servlet.ServletContextAttributeEvent;
      import javax.servlet.ServletContextAttributeListener;
      public class MyServletContextAttributeListener implements ServletContextAttributeListener{
          @Override
          public void attributeAdded(ServletContextAttributeEvent arg0) {
              //添加属性时的监听方法
              System.out.println(arg0.getName());//获得放到域中的name
              System.out.println(arg0.getValue());//获得放到域中的value
          }
          @Override
          public void attributeRemoved(ServletContextAttributeEvent arg0) {
              //移除属性时的监听方法
              System.out.println(arg0.getName());//删除的域中的name
              System.out.println(arg0.getValue());//删除的域中的value
          }
          @Override
          public void attributeReplaced(ServletContextAttributeEvent arg0) {
              //修改属性时的监听方法
              System.out.println(arg0.getName());//获得修改前的域中的name
              System.out.println(arg0.getValue());//获得修改前的域中的value
          }
      }
      
      //web.xml
      <?xml version="1.0" encoding="UTF-8"?>
      <web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://java.sun.com/xml/ns/javaee" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" id="WebApp_ID" version="2.5">
        <listener>
          <listener-class>attribute.MyServletContextAttributeListener</listener-class>
        </listener>
        <display-name>WEB23_LISTENER</display-name>
        <welcome-file-list>
          <welcome-file>index.jsp</welcome-file>
        </welcome-file-list>
        <servlet>
          <description></description>
          <display-name>TestMyServletContextAttributeListener</display-name>
          <servlet-name>TestMyServletContextAttributeListener</servlet-name>
          <servlet-class>attribute.TestMyServletContextAttributeListener</servlet-class>
        </servlet>
        <servlet-mapping>
          <servlet-name>TestMyServletContextAttributeListener</servlet-name>
          <url-pattern>/test1</url-pattern>
        </servlet-mapping>
      </web-app>
      
      //用于测试的接口TestMyServletContextAttributeListener.java
      package attribute;
      import java.io.IOException;
      import javax.servlet.ServletContext;
      import javax.servlet.ServletException;
      import javax.servlet.http.HttpServlet;
      import javax.servlet.http.HttpServletRequest;
      import javax.servlet.http.HttpServletResponse;
      public class TestMyServletContextAttributeListener extends HttpServlet {
          public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
              ServletContext context = this.getServletContext();
              //向context域中存数据
              context.setAttribute("name", "tom");
              //改context数据
              context.setAttribute("name", "lucy");
              //删context数据
              context.removeAttribute("name");
          }
          public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
              doGet(request, response);
          }
      }

      当调用context.setAttribute(“name”, “tom”)时打印name tom;当调用context.setAttribute(“name”, “lucy”)时打印name tom;当调用context.removeAttribute(“name”)时打印name lucy。

    • HttpSessionAttributeListener监听器(同上)

    • ServletRequestAriibuteListenr监听器(同上)

    5.与session中的绑定的对象相关的监听器(对象感知监听器)

    • 即将要被绑定到session中的对象有几种状态

      • 绑定状态:就一个对象被放到session域中(setAttribute)

      • 解绑状态:就是这个对象从session域中移除了(removeAttribute)

      • 钝化状态:是将session内存中的对象持久化(序列化)到磁盘

      • 活化状态:就是将磁盘上的对象再次恢复到session内存中(注意对象必须实现Serializable

      接口)

    • 绑定与解绑的监听器HttpSessionBindingListener(绑在对象上的,且不用在web.xml配置

      • 流程:创建对象实现HttpSessionBindingListener 接口中的public void valueBound(HttpSessionBindingEvent arg0)(绑定的方法,将对象放到session时触发)和public void valueUnbound(HttpSessionBindingEvent arg0)(解绑的方法,将对象从session中移除时触发)。

      • 实例代码:

      //Person.java创建Person类
      package domian;
      import javax.servlet.http.HttpSessionBindingEvent;
      import javax.servlet.http.HttpSessionBindingListener;
      public class Person implements HttpSessionBindingListener{
          private String id;
          private String name;
          public String getId() {
              return id;
          }
          public void setId(String id) {
              this.id = id;
          }
          public String getName() {
              return name;
          }
          public void setName(String name) {
              this.name = name;
          }
          @Override
          public String toString() {
              return "Person [id=" + id + ", name=" + name + "]";
          }
          public Person(String id, String name) {
              super();
              this.id = id;
              this.name = name;
          }
          public Person() {
              super();
          }
          @Override
          public void valueBound(HttpSessionBindingEvent arg0) {
              // 绑定的方法,将person对象放到session时触发
              System.out.println("Person被绑定");
          }
          @Override
          public void valueUnbound(HttpSessionBindingEvent arg0) {
              // 解绑的方法,将person对象从session移除时触发
              System.out.println("Person被解绑");
          }
      }
      
      //TestPersonBindingServlet.java
      package domian;
      import java.io.IOException;
      import javax.servlet.ServletException;
      import javax.servlet.http.HttpServlet;
      import javax.servlet.http.HttpServletRequest;
      import javax.servlet.http.HttpServletResponse;
      import javax.servlet.http.HttpSession;
      public class TestPersonBindingServlet extends HttpServlet {
          public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
              HttpSession session = request.getSession();
              //将person对象绑定到session中
              Person p = new Person("100","jack");
              session.setAttribute("person",p);
              //将person对象从session中解绑
              session.removeAttribute("person");
          }
          public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
              doGet(request, response);
          }
      }

      当访问接口时,代码执行到session.setAttribute("person",p);时输出Person被绑定;代码执行到session.removeAttribute("person");时输出Person被解绑。

    • 钝化与活化的监听器HttpSessionActivationListener 【重要:用于服务器优化】

      • 默认:实现了HttpSessionActivationListener(覆盖其中的public void sessionDidActivate(HttpSessionEvent arg0)public void sessionWillPassivate(HttpSessionEvent arg0)方法)和Serializable接口(注意必须实现Serializable接口)的对象,被放入到session后,当服务器stop时,session会被钝化到work/catalina/localhost中命名为SESSIONS.ser;当服务器start时,session会被活化到session域中。

      • 手动设置钝化时间和存储位置(通过配置文件指定对象钝化时间—对象多长时间不用被钝化):

        • 在META-INF下创建一个context.xml

        • context.xml中的代码:

        <?xml version="1.0" encoding="UTF-8"?>
        <Context>
        <!-- maxIdleSwap:session中的对象多长时间不使用就钝化(单位:分钟),注意钝化和销毁的概念完全不同 -->
        <!-- directory:钝化后的对象的文件写到磁盘的哪个目录下配置钝化的对象文件在 work/catalina/localhost/钝化文件 -->
        <Manager className="org.apache.catalina.session.PersistentManager" maxIdleSwap="1"><!-- Manager是处理内容的对象 -->
            <Store className="org.apache.catalina.session.FileStore" directory="storeFile" /><!-- Store是做存储的对象 -->
        </Manager>
        </Context>
      • 实例代码:

      //Customer.java
      package domian;
      import java.io.Serializable;
      import javax.servlet.http.HttpSessionActivationListener;
      import javax.servlet.http.HttpSessionEvent;
      public class Customer implements HttpSessionActivationListener,Serializable{
          private String id;
          private String name;
          public String getId() {
              return id;
          }
          public void setId(String id) {
              this.id = id;
          }
          public String getName() {
              return name;
          }
          public void setName(String name) {
              this.name = name;
          }
          @Override
          public String toString() {
              return "Customer [id=" + id + ", name=" + name + "]";
          }
          public Customer(String id, String name) {
              super();
              this.id = id;
              this.name = name;
          }
          public Customer() {
              super();
          }
          @Override
          public void sessionDidActivate(HttpSessionEvent arg0) {
              //活化
              System.out.println("customer被活化了");
          }
          @Override
          public void sessionWillPassivate(HttpSessionEvent arg0) {
              //钝化
              System.out.println("customer被钝化了");
          }
      }
      
      //TestCustomerActiveServlet.java 用于将customer对象放至session中
      package domian;
      import java.io.IOException;
      import javax.servlet.ServletException;
      import javax.servlet.http.HttpServlet;
      import javax.servlet.http.HttpServletRequest;
      import javax.servlet.http.HttpServletResponse;
      import javax.servlet.http.HttpSession;
      public class TestCustomerActiveServlet extends HttpServlet {
          public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
              HttpSession session = request.getSession();
              //将customer放到session中
              Customer customer = new Customer("200","lucy");
              session.setAttribute("customer", customer);
              System.out.println("customer被放到session中");
          }
          public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
              doGet(request, response);
          }
      }
      
      //TestCustomerActiveServlet2 用于测试session对象是否被活化
      package domian;
      import java.io.IOException;
      import javax.servlet.ServletException;
      import javax.servlet.http.HttpServlet;
      import javax.servlet.http.HttpServletRequest;
      import javax.servlet.http.HttpServletResponse;
      import javax.servlet.http.HttpSession;
      public class TestCustomerActiveServlet2 extends HttpServlet {
          public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
              HttpSession session = request.getSession();
              //从session域中获取customer
              Customer customer = (Customer)session.getAttribute("customer");
              System.out.println(customer.getName());
          }
          public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
              doGet(request, response);
          }
      }
    • 面试题:当用户很多时,怎样对服务器进行优化?

      通过session的钝化与活化,将长期不操作的session内存中的对象持久化到磁盘,为服务器腾出空间,等到用户再次开始操作时将磁盘上的对象恢复到session内存中。

    二.邮箱服务器

    1.邮箱服务器的基本概念

    • 邮件的客户端:可以只安装在电脑上的也可以是网页形式的

    • 邮件服务器:起到邮件的接受与推送的作用

    • 邮件发送的协议:

      • 协议:就是数据传输的约束
      • 接受邮件的协议:POP3、 IMAP
      • 发送邮件的协议:SMTP
    • 邮箱服务器地址

    这里写图片描述

    2.邮件发送过程

    这里写图片描述

    ​ 过程描述:如果想利用163客户端发送邮件给腾讯邮箱用户,需要先发送给网易邮箱服务器(采用SMTP协议),网易邮箱服务器的SMTP推送给腾讯邮箱服务器的SMTP协议,腾讯邮箱服务器的SMTP将接受到的邮件存储到一个公共区域。当腾讯客户端用户想要接收邮件时,需要发请求给腾讯邮箱服务器(采用POP3协议),该请求会到公共区域取邮件,返回给用户。

    3.邮箱服务器的安装

    • 安装使用易邮邮件服务器
    • 工具 —> 服务器设置 ->设置单域名如jack.com,即用户名为:用户名@jack.com
    • 新账号 —> 输入账号和密码 —> 点击确认即创建用户

    4.邮箱客户端的安装

    • 安装Foxmail
    • 邮箱 —> 新建邮箱账户 —> 写上方的电子邮件及密码并点击next —> 设置接受邮件服务器ip和发送ip均为localhost(本机测试使用)
    • 新建两个用户可以互相发邮件,接受邮件即可
    • 注意:想要接受时需要点击收邮件,因为客户端不是实时刷新的,是固定时间一刷新

    5.使用程序发送邮件

    • 使用mail.jar,内部就有发邮件的API

    • 使用MailUtils.java工具类实现发邮件的功能【开发时直接使用即可,不需要背】

      • 邮件发送流程
        • 创建一个程序与邮件服务器会话对象Session
        • 创建一个Message,它相当于是邮件内容
        • 创建Transport用于将邮件发送
      • 工具代码:
      package mail;
      import java.util.Properties;
      import javax.mail.Authenticator;
      import javax.mail.Message;
      import javax.mail.MessagingException;
      import javax.mail.PasswordAuthentication;
      import javax.mail.Session;
      import javax.mail.Transport;
      import javax.mail.internet.AddressException;
      import javax.mail.internet.InternetAddress;
      import javax.mail.internet.MimeMessage;
      import javax.mail.internet.MimeMessage.RecipientType;
      //邮件工具包
      public class MailUtils {
      //email:邮件发给谁(收件人)  subject:主题  emailMsg:邮件内容
      public static void sendMail(String email,String subject, String emailMsg)
              throws AddressException, MessagingException {
          // 1.创建一个程序与邮件服务器会话对象 Session
          Properties props = new Properties();
          props.setProperty("mail.transport.protocol", "SMTP");//发邮件的协议SMTP
          props.setProperty("mail.host", "192.168.1.109");//发送邮件的服务器地址
          props.setProperty("mail.smtp.auth", "true");// 是否要验证,指定验证为true
          // 创建验证器
          Authenticator auth = new Authenticator() {
              public PasswordAuthentication getPasswordAuthentication() {
                  return new PasswordAuthentication("sam", "111111");//发邮件的账号验证
              }
          };
          //此处的session是会话对象
          Session session = Session.getInstance(props, auth);
          // 2.创建一个Message,它相当于是邮件内容
          Message message = new MimeMessage(session);
          message.setFrom(new InternetAddress("sam@wangzhe.com")); // 设置发送者
          message.setRecipient(RecipientType.TO, new InternetAddress(email)); // 设置发送方式与接收者
          message.setSubject(subject);
          message.setContent(emailMsg, "text/html;charset=utf-8");
          // 3.创建Transport用于将邮件发送
          Transport.send(message);
      }
      }
    • 调用邮件工具类发送邮件

      • 实例代码:
      package mail;
      import javax.mail.MessagingException;
      import javax.mail.internet.AddressException;
      public class SendMailTest {
      public static void main(String[] args) throws AddressException, MessagingException {
          MailUtils.sendMail("jack@wangzhe.com", "测试邮件", "这是一封测试邮件");
      }
      }

    三.案例(定时发送生日祝福)

    • Demo应用jar包

      • c3p0-0.9.1.2.jar
      • commons-dbutils-1.4.jar
      • mail.jar
      • mysql-connector-java-5.0.4-bin.jar
    • 项目思路:

      • 在web应用创建后,当日12点开启任务调度,实现给当天生日的用户发送祝福邮件(本项目作为测试阶段,设置自web应用创建后,每隔10秒钟给当日生日的用户发送祝福邮件)
      • 遍历数据库查看当天过生日的人,借助mail.jar提供的API和MailUtils工具发送邮件
    • 代码编写流程:

      • 创建BirthdayListener类实现ServletContextListener,重写对应方法,主要添加contextInitialized()方法中的内容,利用Timer对象的scheduleAtFixedRate()方法开启任务调度,设置第一个参数为调度任务内容,分别是查询数据库当日过生日的人,发送邮件;设置第二个参数为当前时刻,即项目创建时刻起开启任务调度;设置第三个参数为间隔时长(单位毫秒)
      package birthday;
      import java.sql.SQLException;
      import java.text.SimpleDateFormat;
      import java.util.Date;
      import java.util.List;
      import java.util.Timer;
      import java.util.TimerTask;
      import javax.mail.MessagingException;
      import javax.mail.internet.AddressException;
      import javax.servlet.ServletContextEvent;
      import javax.servlet.ServletContextListener;
      import org.apache.commons.dbutils.QueryRunner;
      import org.apache.commons.dbutils.handlers.BeanListHandler;
      import mail.MailUtils;
      public class BirthdayListener implements ServletContextListener{
      @Override
      public void contextDestroyed(ServletContextEvent arg0) {
          //销毁
      }
      @Override
      public void contextInitialized(ServletContextEvent arg0) {
          //当web应用创建开启任务调度——功能在用户生日当天发送邮件
          Timer timer = new Timer();
          timer.scheduleAtFixedRate(new TimerTask() {
              @Override
              public void run() {
                  // 为当天过生日的用户发送邮件
                  //1.获得今天过生日的人
                  SimpleDateFormat format = new SimpleDateFormat("MM-dd");
                  String currentDate = format.format(new Date());
                  System.out.println(currentDate);
                  //根据当前时间到数据库查今天过生日的人
                  QueryRunner runner = new QueryRunner(DataSourceUtils.getDataSource());
                  String sql = "select * from customer where birthday like ?";
                  List<Customer>customerList = null;
                  try {
                      customerList = runner.query(sql, new BeanListHandler<Customer>(Customer.class), "%"+currentDate+"%");
                  } catch (SQLException e) {
                      // TODO Auto-generated catch block
                      e.printStackTrace();
                  }
                  //2.发邮件
                  if(customerList!=null&&customerList.size()>0) {
                      for(Customer c :customerList) {
                          String emailMsg = "亲爱的:"+c.getRealname()+"生日快乐!";
                          try {
                              MailUtils.sendMail(c.getEmail(), "生日祝福", emailMsg);
                              System.out.println(c.getRealname()+"邮件发送完毕");
                          } catch (AddressException e) {
                              // TODO Auto-generated catch block
                              e.printStackTrace();
                          } catch (MessagingException e) {
                              // TODO Auto-generated catch block
                              e.printStackTrace();
                          }
                      }
                  }
      
              }
          }, new Date(), 1000*10);
          //实际开发中起始时间是一个固定时间
          //实际开发中间隔时间是1天
      }
      }
    • 开发时应用到Customer实体类进行映射,MailUtils工具发送邮件,web.xml配置监听器

      //Customer.java
      package birthday;
      public class Customer {
      private int id;
      private String username;
      private String password;
      private String realname;
      private String birthday;
      private String email;
      public int getId() {
          return id;
      }
      public void setId(int id) {
          this.id = id;
      }
      public String getUsername() {
          return username;
      }
      public void setUsername(String username) {
          this.username = username;
      }
      public String getPassword() {
          return password;
      }
      public void setPassword(String password) {
          this.password = password;
      }
      public String getRealname() {
          return realname;
      }
      public void setRealname(String realname) {
          this.realname = realname;
      }
      public String getBirthday() {
          return birthday;
      }
      public void setBirthday(String birthday) {
          this.birthday = birthday;
      }
      public String getEmail() {
          return email;
      }
      public void setEmail(String email) {
          this.email = email;
      }
      @Override
      public String toString() {
          return "Customer [id=" + id + ", username=" + username + ", password=" + password + ", realname=" + realname
                  + ", birthday=" + birthday + ", email=" + email + "]";
      }
      public Customer(int id, String username, String password, String realname, String birthday, String email) {
          super();
          this.id = id;
          this.username = username;
          this.password = password;
          this.realname = realname;
          this.birthday = birthday;
          this.email = email;
      }
      public Customer() {
          super();
      }
      }
      
      //MailUtils.java
      package mail;
      import java.util.Properties;
      import javax.mail.Authenticator;
      import javax.mail.Message;
      import javax.mail.MessagingException;
      import javax.mail.PasswordAuthentication;
      import javax.mail.Session;
      import javax.mail.Transport;
      import javax.mail.internet.AddressException;
      import javax.mail.internet.InternetAddress;
      import javax.mail.internet.MimeMessage;
      import javax.mail.internet.MimeMessage.RecipientType;
      //邮件工具包
      public class MailUtils {
      //email:邮件发给谁(收件人)  subject:主题  emailMsg:邮件内容
      public static void sendMail(String email,String subject, String emailMsg)
              throws AddressException, MessagingException {
          // 1.创建一个程序与邮件服务器会话对象 Session
          Properties props = new Properties();
          props.setProperty("mail.transport.protocol", "SMTP");//发邮件的协议SMTP
          props.setProperty("mail.host", "192.168.1.109");//发送邮件的服务器地址
          props.setProperty("mail.smtp.auth", "true");// 是否要验证,指定验证为true
          // 创建验证器
          Authenticator auth = new Authenticator() {
              public PasswordAuthentication getPasswordAuthentication() {
                  return new PasswordAuthentication("sam", "111111");//发邮件的账号验证
              }
          };
          //此处的session是会话对象
          Session session = Session.getInstance(props, auth);
          // 2.创建一个Message,它相当于是邮件内容
          Message message = new MimeMessage(session);
          message.setFrom(new InternetAddress("sam@wangzhe.com")); // 设置发送者
          message.setRecipient(RecipientType.TO, new InternetAddress(email)); // 设置发送方式与接收者
          message.setSubject(subject);
          message.setContent(emailMsg, "text/html;charset=utf-8");
          // 3.创建Transport用于将邮件发送
          Transport.send(message);
      }
      }
      
      //web.xml
      <?xml version="1.0" encoding="UTF-8"?>
      <web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://java.sun.com/xml/ns/javaee" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" id="WebApp_ID" version="2.5">
      <listener>
        <listener-class>birthday.BirthdayListener</listener-class>
      </listener>
      <display-name>WEB23_LISTENER</display-name>
      <welcome-file-list>
        <welcome-file>index.jsp</welcome-file>
      </welcome-file-list>
      </web-app>
    更多相关内容
  • 主要给大家介绍了关于java监听器实现和原理的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用java具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
  •  2.1Java的时间监听机制涉及到三个组件:事件源、事件监听器、事件对象  2.2当事件源上发生操作时,它会调用事件监听器的一个方法,并且调用这个方法时,会传递事件对象过来  2.3事件监听器是由开
  • 主要为大家详细介绍了java监听器实现在线人数统计,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
  • 代码用java实现,监听斗鱼直播间的所有弹幕信息,进行输出打印,修改直播房间ID就可以进行使用了
  • java监听器demo

    2016-12-27 17:44:58
    一个java监听器的实现demo,即观察者设计模式实现
  • NULL 博文链接:https://wenchangping.iteye.com/blog/1869464
  • 主要介绍了Java监听器三种实现方法代码解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
  • java监听键盘事件

    2013-06-04 11:06:43
    java开发的监听键盘事件,获取并显示键盘事件
  • java监听数据库

    热门讨论 2013-05-27 09:52:41
    java技术对数据库语句的监听你猜对了万能充
  • 过去使用ASP和ASP.NET两种编程的时候,都写过在线人数统计能,实现功能挺简单的!今天使用java来实现在线人数统计有点另类,是通过Java监听器实现的,需要的朋友可以参考下
  • Java 监听

    千次阅读 2021-03-06 03:35:39
    监听器也叫Listener,是Servlet的监听器,它可以监听客户端的请求、服务端的操作等。通过监听器,可以自动激发一些操作,比如监听在线的用户的数量。例:ServletContextListener监听ServletContext对象。新建一个web...

    监听器也叫Listener,是Servlet的监听器,它可以监听客户端的请求、服务端的操作等。通过监听器,可以自动激发一些操作,比如监听在线的用户的数量。

    例:ServletContextListener监听ServletContext对象。

    新建一个web项目

    2873f4861b6587b2e0fb98f782eefb91.png

    创建CountUtils和ListenerTest类并编写代码

    271454eb089584eaf4fd8baba1378373.png

    ae71124c95988d41f0d8fda65e82b95f.png

    配置web.xml

    b570c036f62e0f2d7d4caba4fe2866a5.png

    编写jsp页面

    e8403a7ef4c373458cfe3552960714c3.png

    226a3fcdefea431a49717e03793c6b8f.png

    运行代码并访问(两个人同时访问)

    992bf83688d6be481e4d34e81e8a7da8.png

    常用的监听接口:

    ServletContextAttributeListener监听对ServletContext属性的操作,比如增加、删除、修改属性。

    ServletContextListener监听ServletContext对象。

    当创建ServletContext时,激发contextInitialized(ServletContextEvent   sce)方法;

    当销毁ServletContext时,激发contextDestroyed(ServletContextEvent   sce)方法。

    HttpSessionListener监听HttpSession的操作:

    当创建一个Session时,激发sessionCreated(HttpSessionEvent   se)方法;

    当销毁一个Session时,激发sessionDestroyed(HttpSessionEvent   se)方法。

    HttpSessionAttributeListener监听HttpSession中的属性的操作:

    当在Session增加一个属性时,激发attributeAdded(HttpSessionBindingEvent  se)方法;

    当在Session删除一个属性时,激发attributeRemoved(HttpSessionBindingEvent   se)方法;

    当在Session属性被重新设置时,激发attributeReplaced(HttpSessionBindingEvent   se)   方法。

    展开全文
  • java监听者模式三个实例

    热门讨论 2013-02-02 21:18:58
    三个有关监听者的实例,一定要理解透哦。 如果工作两年,还没开始学设计模式,说明你out了哦
  • java监听器之定时器

    2018-05-22 10:13:14
    监听器之定时器,写一个定时器的监听器,这个监听器每十秒向控制台输出一次时间信息 博客地址:https://blog.csdn.net/qq_36631076/article/details/80401609
  • Java监听

    千次阅读 2020-12-09 20:26:20
    编写java类实现监听器的接口,并且实现其接口方法 在web.xml文件中对实现的监听器类进行注册 监听器的实现过程: 1.编写java类实现监听器的接口,并且实现其接口方法 不同接口的实现描述 ...

    定义

    • 是Servlet规范定义的一种特殊类
    • 用于监听ServletContext,HttpSession,ServletRequest等域对象的创建,销毁及其属性修改发生变化的事件
    • 在时间发生前后进行一些必要的操作

    实现步骤

    1. 编写java类实现监听器的接口,并且实现其接口方法
    2. 在web.xml文件中对实现的监听器类进行注册

    监听器的实现过程:

    1.编写java类实现监听器的接口,并且实现其接口方法 

    不同接口的实现描述

    • ServletContextListener:监听ServletContext对象应用程序的创建和销毁,ServletContext生命周期的监听
    • HttpSessionListener:对会话对象的创建和销毁的监听
    • ServletRequestListener:对请求对象的创建和销毁的监听
    • ServletContextAttributeListener:对Servlet属性做监听
    • HttpSessionAttributeListener:对会话属性的变化事件进行监听
    • ServletRequestAttributeListener:对请求对象的属性进行监听
    • HttpSessionBindingListener:绑定到session中的某个对象的状态事件监听器(不需要在web.xml中进行绑定)

    该接口有两个方法:

    • contextInitialized():完成对象初始化的监听,web程序启动时
    • contextDestroyed():对web程序的移除销毁,清理工作,防止资源的浪费

    2.在web.xml文件中对实现的监听器类进行注册

    运行看一看:可以看到web应用程序的创建和销毁过程


    HttpSessionListener:对会话对象的创建和销毁的监听

    先创建会话,然后会在1分钟后销毁创建id 

    监听器案例

    通过监听器实现web应用重复登陆,保证在同一个时间只有一个用户在进行登陆

    • 通过过滤器实现登陆控制,未登陆用户不能访问系统首页
    • 用户登录,将登录名存储到session中
    • 登录监听器监听session属性中登录属性的变化
    • 若登录用户用户名已经登录系统,清楚前次的登录信息

    登录页面的设置

    首先在login.jsp页面获取我们的用户值

    然后将这个用户值显示到main.jsp页面中

     

    运行看一看效果! 

    登录权限过滤(由过滤器来实现)

    • 首先创造一个过滤器,并且进行实现

    • 记得需要在web.xml中进行映射的配置

    •  最后在index.jsp页面中,完成弹窗的响应

    • 运行看一看,当我们直接访问main.jsp页面时,会跳出弹窗,并且点击确定后,会自动重定向到登录页面 

    用户信息存储实现(单例模式)

    数据缓存,来存储用户的登录信息,进行登录信息的判断,查看是否有同样的数据进行了登录。

    • 首先,创建一个类,并将它修改为单例模式

    HttpSession对象在创建后,都会有一个唯一的ID用于与其他的session做标志区别

                                                                    

    功能类的总的实现代码:

    package com.kilig.cache;
    
    import java.util.HashMap;
    import java.util.Map;
    
    import javax.servlet.http.HttpSession;
    
    public class LoginCache {
    	//首先将类修改为单例模式的类
    	//1.构造器的私有化
    	//2.定义静态变量
    	private static LoginCache instance =new LoginCache();
    	//定义集合存储缓存信息,首先将用户登录名和SessionID进行关联
    	private Map<String, String> loginUserSession =new HashMap<String, String>();
    	//在存储SessionID和session会话本身映射集合
    	private Map<String, HttpSession> loginSession =new HashMap<String, HttpSession>();
    	private LoginCache() {
    
    	}
    	//3.定义静态方法获取对象的实例
    	public static LoginCache getInstance() {
    		return instance;
    	}
    	//封装方法,操纵集合
    	public String getSessionIdByUsername(String username) {
    		return loginUserSession.get(username);//通过用户名获取集合中对应的NID
    	}
    	public HttpSession getSessionBySessionId(String sessionId) {
    		return loginSession.get(sessionId);//通过SessionId获得Session对象
    	}
    	//存储功能处理
    	public void setSessionIdByUsername(String username, String sessionId) {
    		loginUserSession.put(username, sessionId);//存储username,sessionId到会话缓存中
    	}
    	public void setSessionBySessionId(String sessionId, HttpSession session) {
    		loginSession.put(sessionId, session);//存储sessionId,session到会话缓存中
    	}
    }
    

    监听功能实现

    当session属性创建后,就是用户第一次访问web应用。用户可能只访问了登录界面,而并没有进行登录操作,此时监听到了会话的创建,但是对登录操作来说是没有任何意义的。

    因此我们采用的是属性的监听器。此时我们监听的就是loginUser,当user被创建时,此时就表示我们的用户进行了登录。因此就使用HttpSessionAttributeListener。

    记得在web.xml中进行注册哦 

     

     

     

     

     

     

     

     

     

     

     

    展开全文
  • 主要介绍了Java设计模式之监听器模式,结合实例形式较为详细的分析了java设计模式中监听器模式的概念、原理及相关实现与使用技巧,需要的朋友可以参考下
  • Java监听器详解

    万次阅读 多人点赞 2019-08-17 18:47:42
    一.监听器的定义 ...web监听器由servlet规范提供的,它可以监听客户端的请求,服务端的操作,监听的对象包括ServletContext,HttpSession,ServletRequest三个预对象(内置对象),分别对应aplication,s...

    一.监听器的定义

    是指专门用于对其他对象身上发生的事件或状态改变进行监听和相应处理的对象,当被监视的对象发生变化时,立即采取相应的行动。

    Web应用当中监听器是什么?

    在这里插入图片描述
    web监听器由servlet规范提供的,它可以监听客户端的请求,服务端的操作,监听的对象包括ServletContext,HttpSession,ServletRequest三个预对象(内置对象),分别对应aplication,session,request。
    在这里插入图片描述

    内置对象

    JSP中一共预先定义了9个这样的内置对象,分别为:request、response、session、application、out、pagecontext、config、page、exception
    内置对象(又叫隐含对象)特点:

    1. 由JSP规范提供,不用编写者实例化。
    2. 通过Web容器实现和管理
    3. 所有JSP页面均可使用
    4. 只有在脚本元素的表达式或代码段中才可使用(<%=使用内置对象%>或<%使用内置对象%>)
    request对象

    request 对象是 javax.servlet.httpServletRequest类型的对象。 该对象代表了客户端的请求信息,主要用于接受通过HTTP协议传送到服务器的数据。(包括头信息、系统信息、请求方式以及请求参数等)。request对象的作用域为一次请求。

    session对象

    session 对象是由服务器自动创建的与用户请求相关的对象。服务器为每个用户都生成一个session对象,用于保存该用户的信息,跟踪用户的操作状态。session对象内部使用Map类来保存数据,因此保存数据的格式为 “Key/value”。 session对象的value可以使复杂的对象类型,而不仅仅局限于字符串类型。

    application对象

    application 对象可将信息保存在服务器中,直到服务器关闭,否则application对象中保存的信息会在整个应用中都有效。与session对象相比,application对象生命周期更长,类似于系统的“全局变量”。

    Web监听器概念

    Servlet规范中定义的一种特殊类
    用于监听ServletContext,HttpSession和ServletRequest等域对象的创建与销毁事件用于监听域对象的属性发生修改的事件可以在事件发生前,发生后做一些必要的处理

    二.Web监听器的用途

    1.统计在线人数和在线用户
    2.系统启动时加载初始化信息
    3.统计网站访问量
    4.跟Spring结合

    三.第一个Web监听器

    实现步骤

    1.创建一个实现监听器接口的类

    在这里插入图片描述
    在这里插入图片描述
    代码如下:

    package com.java.listener;
    
    import javax.servlet.ServletContextEvent;
    import javax.servlet.ServletContextListener;
    
    /**
     * @author : xiayj
     * @date : 2019/8/3 17:58
     */
    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");
        }
    }
    
    2.配置web.xml进行注册

    把监听对象放到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">
        <listener>
            <listener-class>com.java.listener.FirstListener</listener-class>
        </listener>
        <welcome-file-list>
            <welcome-file>index.jsp</welcome-file>
        </welcome-file-list>
        <login-config>
            <auth-method>BASIC</auth-method>
        </login-config>
    </web-app>
    

    启动tomcat,控制台输出如下说明当web应用启动时会自动创建一个application对象,且只有一个
    在这里插入图片描述
    关闭tomcat,如下表示当前应用销毁
    在这里插入图片描述

    监听器的启动顺序

    在这里插入图片描述

    四.监听器的分类

    1.按监听的对象划分

    1)用于监听应用程序环境对象(ServletContext)的事件监听器
    2)用于监听用户会话对象(HttpSession)的事件监听器
    3)用于监听请求消息对象(ServletRequest)的事件监听器

    2.按监听的事件划分

    1)监听域对象自身的创建和销毁的事件监听器

    在这里插入图片描述

    ServletContext的创建与销毁

    ServletContext实现了ServletContextListener用于监听它的创建与销毁事件,一个web项目可以定义多个ServletContextListener,但一个web项目只有一个ServletContext对象。
    ServletContextListener有两个事件处理方法,这两个方法都会传入ServletContextEvent事件对象,可以获取ServletContext对象以及一些初始化参数。ServletContextListener主要用途:可以做定时器,加载全局属性对象,创建全局的数据库连接,加载一些缓存信息。
    在这里插入图片描述
    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">
        <listener>
            <listener-class>com.java.listener.FirstListener</listener-class>
        </listener>
        <welcome-file-list>
            <welcome-file>index.jsp</welcome-file>
        </welcome-file-list>
        <login-config>
            <auth-method>BASIC</auth-method>
        </login-config>
    <!--配置初始化参数-->
        <context-param>
            <param-name>initParam</param-name>
            <param-value>listener</param-value>
        </context-param>
    </web-app>
    

    代码实现:

    package com.java.listener;
    
    import javax.servlet.ServletContextEvent;
    import javax.servlet.ServletContextListener;
    
    /**
     * @author : xiayj
     * @date : 2019/8/3 17:58
     */
    public class FirstListener implements ServletContextListener {
        //ServletContext是上下文对象,当web应用启动时创建,web应用销毁时销毁
        @Override
        public void contextInitialized(ServletContextEvent sce) {
    //项目启动时获取初始化参数
            String initParam = sce.getServletContext().getInitParameter("initParam");
            System.out.println("contextInitialized : initParam = " + initParam);
        }
    
        @Override
        public void contextDestroyed(ServletContextEvent sce) {
    //销毁主要用于当数据库连接关闭时释放一些不必要的资源
            System.out.println("contextDestroyed");
        }
    }
    

    项目启动结果如下,说明当项目启动时就加载了在web.xml中配置的参数
    在这里插入图片描述

    HttpSession的创建与销毁

    与ServletContext类似,HttpSession实现了HttpSessionListener用于监听它的创建与销毁事件,一个web项目可以定义多个HttpSessionListener,但一个web项目只有一个HttpSession对象。HttpSessionEvent事件对象,可以获取当前被创建的HttpSession对象。HttpSessionListener主要用途:统计在线人数,记录访问日志。
    在这里插入图片描述
    Session创建:用户打开浏览器第一次访问我们应用时,这次会话web容器会分配一个session,可以在session中保存一些信息
    Session销毁:1.手动点退出,关闭服务器2.关闭浏览器一段时间直到session过期3.不关闭浏览器,session超时
    实现HttpSessionListener:

    package com.java.listener;
    
    import javax.servlet.http.HttpSessionEvent;
    import javax.servlet.http.HttpSessionListener;
    
    /**
     * @author : xiayj
     * @date : 2019/8/4 16:59
     */
    public class MyHttpSessionListener implements HttpSessionListener {
        @Override
        public void sessionCreated(HttpSessionEvent se) {
            System.out.println("sessionCreated");
        }
    
        @Override
        public void sessionDestroyed(HttpSessionEvent se) {
            System.out.println("sessionDestroyed");
        }
    }
    

    web.xml文件加如下配置

    <!--配置session超时时间,1表示1分钟0表示没有超时限制-->
    <session-config>
        <session-timeout>1</session-timeout>
    </session-config>
    

    访问项目
    在这里插入图片描述
    控制台输出如下Listener监听到session创建
    在这里插入图片描述
    等一分钟后,监听到session被销毁
    在这里插入图片描述

    ServletRequest的创建与销毁

    ServletRequest实现了ServletRequestListener用于监听它的创建与销毁事件,一个web项目可以定义多个ServletRequestListener,但一个web项目只有一个ServletRequest对象。ServletRequestListener主要用途:读取request里的参数,记录访问历史,路径。
    在这里插入图片描述
    实现ServletRequestListener

    package com.java.listener;
    
    import javax.servlet.ServletRequestEvent;
    import javax.servlet.ServletRequestListener;
    
    /**
     * @author : xiayj
     * @date : 2019/8/4 17:54
     */
    public class MyServletRequestListener implements ServletRequestListener {
        @Override
        public void requestInitialized(ServletRequestEvent sre) {
            //获取当前的请求参数
            String name = sre.getServletRequest().getParameter("name");
            System.out.println("requestInitialized");
        }
    
        @Override
        public void requestDestroyed(ServletRequestEvent sre) {
            System.out.println("requestDestroyed");
        }
    }
    

    Web.xml中增加监听器配置信息

    <listener>
        <listener-class>com.java.listener.MyServletRequestListener</listener-class>
    </listener>
    

    启动项目,打开浏览器,查看控制台如下,可知request监听到了用户的访问请求,并在请求结束后销毁
    在这里插入图片描述

    2)监听域对象中的属性的增加和删除的事件监听器

    在这里插入图片描述
    代码实现:
    新建三个类

    package com.java.listener;
    
    import javax.servlet.ServletRequestAttributeEvent;
    import javax.servlet.ServletRequestAttributeListener;
    
    /**
     * @author : xiayj
     * @date : 2019/8/10 17:18
     */
    public class MyServletRequestAttributeListener implements ServletRequestAttributeListener {
        @Override
        public void attributeAdded(ServletRequestAttributeEvent srae) {
    //可知是什么对象添加了什么属性,移除了什么属性       System.out.println("ServletRequest_attributeAdded:"+srae.getName());
        }
    
        @Override
        public void attributeRemoved(ServletRequestAttributeEvent srae) {
            System.out.println("ServletRequest_attributeRemoved:"+srae.getName());
        }
    
        @Override
        public void attributeReplaced(ServletRequestAttributeEvent srae) {
            System.out.println("ServletRequest_attributeReplaced:"+srae.getName());
        }
    }
    
    package com.java.listener;
    
    import javax.servlet.http.HttpSessionAttributeListener;
    import javax.servlet.http.HttpSessionBindingEvent;
    
    /**
     * @author : xiayj
     * @date : 2019/8/10 16:55
     */
    public class MyHttpSessionAttributeListener implements HttpSessionAttributeListener {
        @Override
        public void attributeAdded(HttpSessionBindingEvent se) {
            System.out.println("HttpSession_attributeAdded:"+se.getName());
        }
    
        @Override
        public void attributeRemoved(HttpSessionBindingEvent se) {
            System.out.println("HttpSession_attributeRemoved:"+se.getName());
        }
    
        @Override
        public void attributeReplaced(HttpSessionBindingEvent se) {
            System.out.println("HttpSession_attributeReplaced:"+se.getName());
        }
    }
    
    package com.java.listener;
    
    import javax.servlet.ServletContextAttributeEvent;
    import javax.servlet.ServletContextAttributeListener;
    
    /**
     * @author : xiayj
     * @date : 2019/8/10 16:52
     */
    public class MyServletContextAttributeListener implements ServletContextAttributeListener {
        @Override
        public void attributeAdded(ServletContextAttributeEvent scae) {
            System.out.println("ServletContext_attributeAdded:"+scae.getName());
        }
    
        @Override
        public void attributeRemoved(ServletContextAttributeEvent scae) {
            System.out.println("ServletContext_attributeRemoved:"+scae.getName());
        }
    
        @Override
        public void attributeReplaced(ServletContextAttributeEvent scae) {
            System.out.println("ServletContext_attributeReplaced:"+scae.getName());
        }
    }
    

    Web.xml文件中注册监听器

    <listener>
        <listener-class>com.java.listener.MyHttpSessionAttributeListener</listener-class>
    </listener>
    <listener>
        <listener-class>com.java.listener.MyServletContextAttributeListener</listener-class>
    </listener>
    <listener>
        <listener-class>com.java.listener.MyServletRequestAttributeListener</listener-class>
    </listener>
    

    创建新增各个属性值得jsp界面,为了方便在每个页面都加上了初始化,销毁的按钮
    index.jsp

    <%@ page contentType="text/html;charset=UTF-8" language="java" %>
    <html>
      <head>
        <title>这是myweb应用</title>
      </head>
      <body>
      你好!这是一个listener
      <button onclick="location.href='<%=request.getContextPath()%>/init.jsp';">Init</button>
      <button onclick="location.href='<%=request.getContextPath()%>/destory.jsp';">Destory</button>
      </body>
    </html>
    

    init.jsp

    <%@ page contentType="text/html;charset=UTF-8" language="java" %>
    <%
    String path = request.getContextPath();
    String basePath = request.getScheme()+"://"+request.getServletPath();
    //添加属性,让Listener监听执行attributeAdded方法
    request.setAttribute("requestName","requestValue");
    request.getSession().setAttribute("sessionName","sessionValue");
    request.getSession().getServletContext().setAttribute("contextName","contextValue");
    %>
    <html>
    <head>
        <title>Title</title>
    </head>
    <body>
    这是初始化值得界面
    <button onclick="location.href='<%=request.getContextPath()%>/init.jsp';">Init</button>
    <button onclick="location.href='<%=request.getContextPath()%>/destory.jsp';">Destory</button>
    </body>
    </html>
    

    destory.jsp

    <%@ page contentType="text/html;charset=UTF-8" language="java" %>
    <%
        String path = request.getContextPath();
        String basePath = request.getScheme()+"://"+request.getServletPath();
        //移除属性,让Listener监听执行attributeRemoved方法
        request.removeAttribute("requestName");
        request.getSession().removeAttribute("sessionName");
        request.getSession().getServletContext().removeAttribute("contextName");
    %>
    <html>
    <head>
        <title>Title</title>
    </head>
    <body>
    这是销毁界面
    <button onclick="location.href='<%=request.getContextPath()%>/init.jsp';">Init</button>
    <button onclick="location.href='<%=request.getContextPath()%>/destory.jsp';">Destory</button>
    </body>
    </html>
    

    超时时间改为0

    <!--配置session超时时间,1表示1分钟0表示没有超时限制-->
    <session-config>
        <session-timeout>0</session-timeout>
    </session-config>
    

    打开主页面
    在这里插入图片描述
    点击init,控制台输出如下
    在这里插入图片描述
    再次点击init
    在这里插入图片描述
    点击destory,属性被移除,request对象因为已经被销毁了,域中已没有requestName属性,所以没有移除信息
    在这里插入图片描述

    3)监听绑定到HttpSession域中的某个对象的状态的事件监听器

    在这里插入图片描述
    绑定:通过getSession().setAttribute将对象保存到session对象中
    解除绑定:通过getSession().removeAttribute将对象从session对象中移除
    钝化:将session对象持久化到存储设备上
    活化:将session对象从存储设备上恢复

    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述
    代码实现:
    新建user类

    package com.java.entity;
    
    import javax.servlet.http.HttpSessionActivationListener;
    import javax.servlet.http.HttpSessionBindingEvent;
    import javax.servlet.http.HttpSessionBindingListener;
    import javax.servlet.http.HttpSessionEvent;
    import java.io.Serializable;
    
    /**
     * @author : xiayj
     * @date : 2019/8/17 16:09
     */
    /** 想要钝化或活化必须实现Serializable接口*/
    public class User implements HttpSessionBindingListener, HttpSessionActivationListener, Serializable {
        private String username;
        private String password;
        /** 对象绑定执行*/
        @Override
        public void valueBound(HttpSessionBindingEvent event) {
            System.out.println("valueBound Name:"+event.getName());
        }
        /** 对象解除绑定执行*/
        @Override
        public void valueUnbound(HttpSessionBindingEvent event) {
            System.out.println("valueUnbound Name:"+event.getName());
        }
        /** 钝化*/
        @Override
        public void sessionWillPassivate(HttpSessionEvent se) {
            System.out.println("sessionWillPassivate"+se.getSource());
        }
        /** 活化*/
        @Override
        public void sessionDidActivate(HttpSessionEvent se) {
            System.out.println("sessionDidActivate"+se.getSource());
        }
    
        public String getUsername() {
            return username;
        }
    
        public void setUsername(String username) {
            this.username = username;
        }
    
        public String getPassword() {
            return password;
        }
    
        public void setPassword(String password) {
            this.password = password;
        }
    }
    
    绑定/解除绑定测试:

    在init文件里增加如下:

    //将用户实例放到session中
    request.getSession().setAttribute("currentUser",new com.java.entity.User());
    

    在destory文件里增加如下:

    request.getSession().removeAttribute("currentUser");
    

    启动项目,打开首页:
    在这里插入图片描述
    点击init和destory,控制台分别输出如下,表明当对象实现HttpSessionBindingListener接口后,只要对象被session绑定,就会触发事件,执行valueBound方法,当解除绑定后,执行valueUnBound方法
    在这里插入图片描述
    在这里插入图片描述

    钝化/活化测试

    启动tomcat,打开项目,可以看到控制台已经创建session
    在这里插入图片描述
    点击init,可以看到session中保存了几个属性值
    在这里插入图片描述
    关闭tomcat,控制输出如下,sessionWillPassivateorg.apache.catalina.session.StandardSessionFacade@1e7c7811 就是session钝化的输出,StandardSession是默认的钝化管理器
    在这里插入图片描述
    钝化的文件保存位置如下
    在这里插入图片描述
    在首页新增如下:

    <!-- 在首页获取用户对象 正常情况下因为tomcat已关闭,session中用户对象应该已被销毁,
    但因为实现了HttpSessionActivationListener接口,tomcat重新启动时会活化保存的session文件-->
    <%=request.getSession().getAttribute("currentUser")%>
    你好!这是一个listener
    <button onclick="location.href='<%=request.getContextPath()%>/init.jsp';">Init</button>
    <button onclick="location.href='<%=request.getContextPath()%>/destory.jsp';">Destory</button>
    

    重启tomcat,控制台输出如下:
    在这里插入图片描述
    打开首页,可以看到之前钝化的user对象被输出
    在这里插入图片描述
    Ps:如果出现可以活化,但是无法钝化的问题即重启tomcat没有调用 sessionDidActivate()方法,浏览器端也获取不到重启前存入的对象
    这是因为每次重启tomcat会将如下目录删除并重新生成,之前钝化的session文件也会被一起删除
    在这里插入图片描述
    解决方法:在tomcat安装目录下conf/context.xml配置文件里增加如下配置,directory里是自定义的session文件存储路径

    <Manager className="org.apache.catalina.session.PersistentManager" saveOnRestart="true">
            <Store className="org.apache.catalina.session.FileStore" directory="D:\tomcat\session"/>
    </Manager>
    

    在Servlet3.0版本以上提供@WebListener注解将一个实现了特定监听器接口(比如ServletContextListener,HttpSessionListener)的类定义为监听器,这样在web应用中使用监听器时不需要在web.xml文件中配置监听器的相关描述信息,感兴趣的可以自己去研究一下,用法差不多。

    参考文章:Jsp:九大内置对象https://my.oschina.net/u/3805464/blog/1813805
    本文章内容是对慕课网课程内容的一个总结,感兴趣的同学可以去慕课网学习该课程
    地址如下:https://www.imooc.com/learn/271

    展开全文
  • java 监听

    千次阅读 2019-07-14 11:32:13
    最开始理解java监听器的时候我有这些疑问! java监听器监听的内容是什么?监听器作用在哪个事件上?是在哪配置的监听内容的? 就带着这些问题理解一下java监听器吧! 首先,监听器要怎么实现? Java事件由事件类...
  • Java 监听

    千次阅读 2019-06-06 17:14:27
    java的设计模式中,有一种模式叫:观察者模式,和这个类似。举个例子,本例子是一个简单的监听当数据发生变化时要做的操作。 1,我们先定义一个接口,可以让多个监听者实现 publicinterfaceIDataListen{ ...
  • 总结:Java监听器本质

    千次阅读 2022-01-21 09:46:05
    监听器是基于事件驱动的,用于对操作事件进行监听,当监听到有操作事件的时候进行的逻辑处理,是观察者设计模式的应用。 2、监听器三元素 事件源:事件发生的源头。比如Zookeeper 事件:对事件进行抽象、封装 ...
  • java监听数据库表变化

    千次阅读 2021-04-23 20:43:28
    南京理工大学泰州科技学院 实验报告书课程名称: 《Java 面向对象程序设计》 实验题目: 实验七 输入输出流,数据库编程 班学姓级: 号: 名: 09 计算机(2) ......一、实验目的与要求 Java 面向对象编程 实验报告 1 、SQL...
  • Java监听器的四种实现方法

    千次阅读 2020-10-21 11:04:04
    监听器的三种实现方法(OnlickListener) android监听有4种实现方法,匿名内部类实现,内部类实现,实现事件接口,覆写方法实现,控件onClick属性实现事件,下面我将一一举例。 欢迎使用Markdown编辑器 你好! 这是你...
  • 本篇文章主要介绍了Java可以如何实现文件变动的监听的示例,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
  • Java监听链上智能合约事件

    千次阅读 2021-07-13 15:19:04
    1、事件的监听 在智能合约中定义的任何事件都将用Event方法在智能合约封装包中进行表示,该事件采用交易收据,并从中提取索引和非索引事件参数。 监听机制是建立在日志基础之上,如果智能合约触发了一个事件,那么...
  • 主要介绍了Java监听器的作用及用法代码示例,具有一定参考价值,需要的朋友可以了解下。
  • java监听Tomcat是否宕机

    2013-12-11 16:35:52
    java监听Tomcat是否宕机 可以重启
  • java监听器练习

    2013-11-18 21:20:06
    java监听器练习,java初学者!
  • 本文中主要陈述一种实时监听文件夹中是否有文件增加的功能,可用于实际文件上传功能的开发。本文通过实例代码给大家介绍的非常详细,具有一定的参考借鉴价值,需要的朋友参考下吧
  • Java监听器的原理和组成及实现

    千次阅读 2021-03-04 03:48:23
    例如:创建一个“人”类Class Person人拥有吃的方法public void eat(){},我们的目的就是,在这个人吃之前要提醒他洗手,所以我们就要监听这个人,不管他在什么时候吃,我们都要能够监听到并且提醒他洗手,通俗点讲这...
  • 下面小编就为大家带来一篇Java线程监听,意外退出线程后自动重启的实现方法。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧

空空如也

空空如也

1 2 3 4 5 ... 20
收藏数 578,132
精华内容 231,252
关键字:

java监听

java 订阅
友情链接: 巡线4.py.zip