精华内容
下载资源
问答
  • Java监听模式

    千次阅读 2018-02-07 22:12:46
    Java监听模式右三个部分组成:事件源、事件对象、事件监听器 当事件源触发某种行为,会自动执行事件监听器里面相应的方法 事件监听器方法参数是事件对象,事件对象包含事件源的引用,在方法中可以获取事件源的引用 ...

    说明

      生活中,监听无处不在。比如说,手机播放音乐功能,也是一种监听:你不点击播放按钮,手机就不放歌,当你点击时,手机就播放音乐。即触发某种行为,便执行相应的动作。

    组成

    1. Java监听模式右三个部分组成:事件源、事件对象、事件监听器
    2. 当事件源触发某种行为,会自动执行事件监听器里面相应的方法
    3. 事件监听器方法参数是事件对象,事件对象包含事件源的引用,在方法中可以获取事件源的引用

    PS:好吧,说的有点晕,因为我也是刚学的啦,看的晕的同学直接看代码就好了。

    实现

    • 事件源:Student拥有read()方法,
    package com.petro.listener;
    /**
     * @ClassName: Student
     * @description: 监听模式的事件源
     * @author Petro Chen
     * @date: 2017年9月12日下午11:32:03
     */
    public class Student {
    
        // read方法事件监听器引用
        private Listener listener;
    
        /**
         * @description: 注册read方法的事件监听器
         * @param listener 事件监听器引用
         * @return void
         * @throws
         */
        public void setReadListener(Listener listener) {
            this.listener = listener;
        }
    
        /**
         * @description: 学生的read方法
         * @return void
         * @throws
         */
        public void read() {
            if (listener != null) {
                Event event = new Event();
                event.setStudent(this);
                // 学生正在读书...
                listener.isReading(event);
            }
        }
    }
    
    • 事件监听器:用来监听Student.read()方法
    package com.petro.listener;
    
    /**
     * @ClassName: Listener
     * @description: 事件监听器
     * @author Petro Chen
     * @date: 2017年9月12日下午11:33:56
     */
    public interface Listener {
        /**
         * @description: 学生读书触发的事件
         * @param event 事件对象引用,通过这个引用,获取事件源的引用,然后就可以对事件源进行操作
         * @return void
         * @throws
         */
        public void isReading(Event event);
    }
    
    • 事件对象:持有事件源的引用
    package com.petro.listener;
    /**
     * @ClassName: Event
     * @description: 事件对象
     * @author Petro Chen
     * @date: 2017年9月12日下午11:41:28
     */
    public class Event {
        // 放置事件源的引用
        private Student student;
    
        public Student getStudent() {
            return student;
        }
    
        public void setStudent(Student student) {
            this.student = student;
        }
    }
    

      到此为止,我们的监听模式写完了,接下来当然是测试啦

    • 测试部分:监听接口实现
    package com.petro.test;
    
    import com.petro.listener.Event;
    import com.petro.listener.Listener;
    /**
     * @ClassName: ReadListener
     * @description: 监听器实现,读书(read方法)监听器
     * @author Petro Chen
     * @date: 2017年9月13日上午12:03:35
     */
    public class ReadListener implements Listener {
    
        @Override
        public void isReading(Event event) {
            /**
             * 使用 event.getStudent()方法获取事件源的引用,然后执行相关操作
             */
            System.out.println("正在读书...");
        }
    
    }
    • 测试部分:程序入口
    package com.petro.test;
    
    import com.petro.listener.Student;
    /**
     * @ClassName: Test
     * @description: 监听器模式测试
     * @author Petro Chen
     * @date: 2017年9月13日上午12:07:04
     */
    public class Test {
        public static void main(String[] args) {
            ReadListener readListener = new ReadListener();
            Student student = new Student();
            student.setReadListener(readListener);
            // 当执行这个方法时,会自动调用ReadListener.isReading()方法
            student.read();
        }
    }
    


    • 监听模式demo地址,有需要的可以去clone

    https://github.com/MiniPeter/listen-mode-demo.git

    结语

      本篇博文到此差不多就结束啦,对本篇博文有建议或者疑惑的可以留言评论哦。欢迎指正错误>_<

    展开全文
  • JAVA监听

    万次阅读 多人点赞 2018-08-15 00:45:49
    作为JAVA开发人员,我们应对javaweb的三个技术要点之一的Listener进行学习,楼主查阅相关资料并学习相关视频,总结如下JAVA监听器内容,供大家参考学习。完整代码内容我已放到我的github中,请自行下载:...

    作为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监听器详解

    千次阅读 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监听键盘事件

    万次阅读 2013-09-11 12:31:50
    简单写一个java Gui键盘监听事件,实现的效果就是按下键盘控制台输出你按下的键。比如:按下A控制台就输出A    效果如图:        以下把实现的效果分为几个步骤:    1.新建一个窗体类继承窗体; ...
    简单写一个java Gui键盘监听事件,实现的效果就是按下键盘控制台输出你按下的键。比如:按下A控制台就输出A 
    

           

            效果如图:

              

     

             

                 以下把实现的效果分为几个步骤:

     

                   1.新建一个窗体类继承窗体;

     

                   2.给这个窗体添加addKeyListener(new MyKeyListener()),方法里面实例化监听键盘类;

     

                   3.新建一个键盘类继承KeyAdapter;

     

                   4.类里面的方法名必须是keyPressed(KeyEvent e);//KeyEvent就是jre对你按下/释放键盘按键的包装

                 

                   5.实例化窗体类。

     

                   下面贴出上图实现的代码:

     

                   

    1. import java.awt.event.KeyAdapter;  
    2. import java.awt.event.KeyEvent;  
    3.   
    4. import javax.swing.JFrame;  
    5.   
    6.   
    7. /** 
    8.  *  
    9.  */  
    10.   
    11. /** 
    12.  * @author Administrator 
    13.  * 
    14.  */  
    15. //窗体类   
    16. public class MyFrame extends JFrame {  
    17.   
    18.     /** 
    19.      * @param args 
    20.      */  
    21.     char charA;  
    22.     public MyFrame(){  
    23.         this.setSize(500,100);  
    24.         this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);  
    25.         this.setTitle("my jframe");  
    26.         this.setVisible(true);  
    27.         this.addKeyListener(new MyKeyListener());  
    28.           
    29.           
    30.     }  
    31.       
    32.       
    33.       
    34.     public static void main(String[] args) {  
    35.         // TODO Auto-generated method stub   
    36.         new MyFrame();  
    37.     }  
    38.   
    39. }  
    40. //监听键盘类   
    41. class MyKeyListener extends KeyAdapter{  
    42.     public void keyPressed(KeyEvent e){  
    43.         char charA=e.getKeyChar();  
    44.         System.out.println("你按了《"+charA+"》键");  
    45.     }  
    46. }  
    import java.awt.event.KeyAdapter;
    import java.awt.event.KeyEvent;
    
    import javax.swing.JFrame;
    
    
    /**
     * 
     */
    
    /**
     * @author Administrator
     *
     */
    //窗体类
    public class MyFrame extends JFrame {
    
    	/**
    	 * @param args
    	 */
    	char charA;
    	public MyFrame(){
    		this.setSize(500,100);
    		this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
    		this.setTitle("my jframe");
    		this.setVisible(true);
    		this.addKeyListener(new MyKeyListener());
    		
    		
    	}
    	
    	
    	
    	public static void main(String[] args) {
    		// TODO Auto-generated method stub
    		new MyFrame();
    	}
    
    }
    //监听键盘类
    class MyKeyListener extends KeyAdapter{
    	public void keyPressed(KeyEvent e){
    		char charA=e.getKeyChar();
    		System.out.println("你按了《"+charA+"》键");
    	}
    }
    


     

     

    另外一种写法就是在给窗体添加监听的时候直接 实例方法,不用再写一个类。建议用这个写法。代码如下:

     

     

    1. import java.awt.event.KeyAdapter;  
    2. import java.awt.event.KeyEvent;  
    3.   
    4. import javax.swing.JFrame;  
    5.   
    6.   
    7. /** 
    8.  *  
    9.  */  
    10.   
    11. /** 
    12.  * @author Administrator 
    13.  * 
    14.  */  
    15. //窗体类   
    16. public class MyFrame extends JFrame {  
    17.   
    18.     /** 
    19.      * @param args 
    20.      */  
    21.     char charA;  
    22.     public MyFrame(){  
    23.         this.setSize(500,100);  
    24.         this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);  
    25.         this.setTitle("my jframe");  
    26.         this.setVisible(true);  
    27.         this.addKeyListener(new KeyAdapter(){  
    28.             public void keyPressed(KeyEvent e){  
    29.                 char charA=e.getKeyChar();  
    30.                 System.out.println("你按了《"+charA+"》键");  
    31.             }  
    32.         });  
    33.           
    34.           
    35.     }  
    36.     public static void main(String[] args) {  
    37.         // TODO Auto-generated method stub   
    38.         new MyFrame();  
    39.     }  
    40.   
    41. }  
    展开全文
  • java监听器的使用技术整理

    千次阅读 2011-03-18 11:33:00
    java监听器的使用技术整理
  • Java监听:一个Java睡前故事(Java Listener: A Java Bedtime Story) 作者: 杨帆 博 邮 本文是仿照 .NET Delegates: A C# Bedtime Story 的功能实现的Java版源代码,时间原因,没有给出文字故事。请参见原文链接。...
  • Java监听端口(一)

    万次阅读 2018-01-24 15:34:00
    比较简单直接上代码,思路见批注 ...import java.io.BufferedReader; import java.io.IOException; import java.io.InputStreamReader; import java.net.ServerSocket; import java.net.Socket; public class Se
  • 一、概念 1.事件监听器 事件监听器就我个人的理解就是:被外部事件(键盘、鼠标)引发的程序...里边有处理事件的方法。从逻辑上看是这样的,但是人家既然这样来命名了,那也没有办法。因为程序员只要知道这么去添加监听
  • Java web应用中,listener监听器是不可或缺的。常用来监听servletContext、httpSession、servletRequest等域对象的创建、销毁以及属性的变化等等,能够在这些事件动作前后进行一定的逻辑处理。 2、应用场景 利用...
  • 调用第三方sdk的方式提供的api,发现api以监听的方式异步返回结果,而我方要求提供实时数据给用户,所以这里就涉及到第三方异步接口同步调用的问题。开始的想法是以简单的while语句循环使用,但这种方式实在不优雅。...
  • java 监听多线程超时:Future

    千次阅读 2018-04-24 17:20:31
    本文将介绍一些针对线程超时的处理方法,希望对初学者能有所帮助 ^_^场景1. 线程意外地在某个while或者哪里陷入死循环了(如果不是意外地请一定先修正逻辑错误),这个线程就这样一直在死循环中,并且随着时间推移...
  • Java监听应用的启动与关闭实例

    千次阅读 2014-02-25 16:59:01
    添加一个监听应用启动和结束监听器这个监听在应用每次启动和结束的时候相应的方法会被容器(tomcat)调用。  1.编写我们的自己的监听类,这个类要实现 ServletContextListener  里面有两个方法,一个在应用开始的...
  • Java结束线程的三种方法

    万次阅读 2018-06-28 14:20:38
    转载自https://blog.csdn.net/xu__cg/article/details/52831127线程属于一次性消耗品,在执行完run()方法之后线程便会正常结束了,线程结束后便会销毁,不能再次start,只能重新建立新的线程对象,但有时run()方法是...
  • java实现监听进程

    千次阅读 2015-01-09 15:59:42
    最近在阿里云搞了一个云服务器 选用的是windows ...所以我就想搞一个程序挂在服务器上监听我的农场外挂,如果外挂自动退出了,我就给自己发一封邮件。 想来想去我决定用Runtime来实现这个功能。 思路如下 1、每隔1分
  • CountDownLatch类位于java.util.concurrent包下,使用CountDownLatch可以实现一个类似计数器的功能。 // 构造参数为计数总量 final CountDownLatch latch = new CountDownLatch(2); 创建一个CountDownLatch的实例...
  • 直接上代码: package test1;...import java.awt.event.*; @SuppressWarnings("serial") public class KeyListenerTest extends JFrame { public KeyListenerTest() { MyWindow mywindow = new...
  • AnimationDrawable监听播放结束,返回到第一张图片~ImageSwitcher动画图片切换,带动画
  • 什么是事件监听? 例如,点击按钮,弹出一个文本框,点击一个文本,文本被删除; 按钮、文本等组件就是事件源 事件就是从开始触发事件源到出现效果的一个过程 以下是一个案例,点击按钮,记录点击的次数和获取...
  • Java Web监听器-学习笔记

    千次阅读 2015-09-18 22:13:37
    一,Web监听器: 是由Servlet规范提供的,可以对客户端的请求、服务端的操作进行监听,包括对象ServletContext、HttpSession、ServletRequest,分别对应application、session、request对象。 application是上下文...
  • java处理一般的文件下载都是直接文件流,可以达到浏览器自动弹出下载窗口的效果,如果是文件比较大或者后台处理逻辑用时比较长,为了页面更加友好,需要添加“遮罩层友情提示”,主要问题就是监听到文件传输完毕,...
  • Java ServerSocket 手动关闭监听

    千次阅读 2016-07-07 20:38:16
    今天练习Java Socket用法的示例代码,发现一些问题 服务器代码: import java.net.*; import java.io.*; public class Test_Socket { public static void main(String[] args)throws IOException { // ...
  • Java-Servlet事件监听

    2012-12-05 20:14:52
    监听器就是一个实现特定接口的普通java程序,这个程序专门用于监听另一个java对象的方法调用或属性改变,当被监听对象发生上述事件监听器某个方法将立即被执行。 监听器典型案例:监听window窗口的事件监听器 ...
  • 监听者 /** * @author 12130 * @date 2019/11/26 * @time 22:35 */ public interface LifeCycleListener { // 当出现某些事件的时候,通知监听者 void onEvent(ObservableRunnable.RunnableEvent event); } ...
  • 这是一个按钮监听(b是按钮,f是JFrame的对象),如果我在里面用Thread.sleep()方法休眠,会出现窗体,但是内容不会显示(休眠结束后窗体就执行最后的语句消失了),后来我把最后的f.setVisible(false);换成了System...
  • 实现方法:使用了两个监听器,一个request用户请求的监听,用于获取用户的信息,一个HttpSession监听,则统计用户的数量 request监听器代码如下: package Litener; import java.text.SimpleDateFormat; import ...
  • Java中的监听

    千次阅读 多人点赞 2013-04-25 16:02:54
     Servlet中的listener的作用是通过监听一些事件的发生,listener在事件发生前后可以做一些必要的处理。   2、原理  简单理解就是事件源生成事件并将其发送至一个或多个监听器,监听器简单地等待,直到它收到一...
  • java web中监听器的使用

    千次阅读 2017-02-04 17:55:34
    java web应用程序在web容器中运行时,在java web应用程序内部会不断发生各种事件,例如web应用的启动,暂停,销毁等。以及web应用中session开始和结束 这些web应用对开发者来说通常是看不见的。其实在servletapi...
  • android监听视频播放结束

    千次阅读 2018-04-07 15:56:50
    android监听视频是否播放结束,今天做的项目是在一个主界面中插入一个播放器,所以不是单纯的视频app,以前用的方法是:mvideoview.setOnCompletionListener(new OnCompletionListener() { @Override public void...
  • /* * @author uv * @date 2018/12/19 17:37 */ import java.nio.file.FileSystems; import java.nio.file.Paths;...import java.nio.file.StandardWatchEventKinds;...import java.nio.file.WatchE...

空空如也

空空如也

1 2 3 4 5 ... 20
收藏数 122,976
精华内容 49,190
关键字:

java监听方法结束后

java 订阅