Listener

1、监听器Listener
javaEE包括13门规范,在课程中主要学习servlet技术和jsp技术。
其中servlet规范包括三个技术点:servlet、listener、filter。

1.1、什么是监听器?

监听器就是监听某个对象的的状态变化的组件。

监听器的相关概念:

事件源:被监听的对象 —– 三个域对象 request、session、servletContext;
监听器:监听事件源对象,事件源对象的状态的变化都会触发监听器 —- 6+2;
注册监听器:将监听器与事件源进行绑定;
响应行为:监听器监听到事件源的状态变化时所涉及的功能代码 —- 程序员编写代码。

1.2、监听器有哪些?

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

这里写图片描述

1.3、监听三大域对象的创建与销毁的监听器

监听ServletContext域的创建与销毁的监听器ServletContextListener

Servlet域的生命周期
何时创建:服务器启动创建
何时销毁:服务器关闭销毁

监听器的编写步骤(重点):
a、编写一个监听器类去实现监听器接口
b、覆盖监听器的方法
c、需要在web.xml中进行配置—注册

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{

    @Override
    //监听context域对象的创建
    public void contextInitialized(ServletContextEvent sce) {
        //就是被监听的对象---ServletContext
        //ServletContext servletContext = sce.getServletContext();

        //getSource就是被监听的对象,是通用的方法
        //ServletContext source = (ServletContext) sce.getSource();

        //System.out.println("context创建了....");

        //开启一个计息任务调度----每天晚上12点 计息一次
        //Timer timer = new Timer();
        //task:任务  firstTime:第一次执行时间  period:间隔执行时间
        //timer.scheduleAtFixedRate(task, firstTime, period);
        /*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 = "2016-08-19 00:00:00";
        String currentTime = "2016-08-18 09:34:00";
        Date parse = null;
        try {
            parse = format.parse(currentTime);
        } catch (ParseException e) {
            e.printStackTrace();
        }

        timer.scheduleAtFixedRate(new TimerTask() {
            @Override
            public void run() {
                System.out.println("银行计息了.....");
            }
        } , parse, 24*60*60*1000);*/

    }

    //监听context域对象的销毁
    @Override
    public void contextDestroyed(ServletContextEvent sce) {
        System.out.println("context销毁了....");
    }
}

web.xml注册

<listener>
    <listener-class>com.attribute.MyServletContextListener</listener-class>
</listener>

ServletContextListener监听器的主要作用

a、初始化的工作:初始化对象,初始化数据 —- 加载数据库驱动,连接池的初始化。
b、加载一些初始化的配置文件 — spring的配置文件
c、任务调度—-定时器—-Timer/TimerTask。

监听Httpsession域的创建于销毁的监听器HttpSessionListener

HttpSession对象的生命周期
何时创建:第一次调用request.getSession时创建
何时销毁:服务器关闭销毁 session过期 手动销毁

public class MyHttpSessionListener implements HttpSessionListener{
    @Override
    public void sessionCreated(HttpSessionEvent se) {
        System.out.println("session创建"+se.getSession().getId());
    }
    @Override
    public void sessionDestroyed(HttpSessionEvent se) {
        System.out.println("session销毁");
    }
}
<listener>
    <listener-class>com.attribute.MyHttpSessionListener</listener-class>
</listener>

监听ServletRequest域创建与销毁的监听器ServletRequestListener

ServletRequest的生命周期
创建:每一次请求都会创建request
销毁:请求结束

1.4、监听三大域对象的属性变化的

域对象的通用的方法:

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

ServletContextAttibuteListener监听器

public class MyServletContextAttributeListener implements ServletContextAttributeListener{
    @Override
    public void attributeAdded(ServletContextAttributeEvent scab) {
        //放到域中的属性
        System.out.println(scab.getName());//放到域中的name
        System.out.println(scab.getValue());//放到域中的value
    }
    @Override
    public void attributeRemoved(ServletContextAttributeEvent scab) {
        System.out.println(scab.getName());//删除的域中的name
        System.out.println(scab.getValue());//删除的域中的value
    }
    @Override
    public void attributeReplaced(ServletContextAttributeEvent scab) {
        System.out.println(scab.getName());//获得修改前的name
        System.out.println(scab.getValue());//获得修改前的value
    }
}
<listener>
    <listener-class>com.attribute.MyServletContextAttributeListener</listener-class>
</listener>
protected 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");
}

HttpSessionAttributeListener监听器(同上)
ServletRequestAriibuteListenr监听器(同上)

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

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

绑定状态:就一个对象被放到session域中
解绑状态:就是这个对象从session域中移除了
钝化状态:是将session内存中的对象持久化(序列化)到磁盘
活化状态:就是将磁盘上的对象再次恢复到session内存中

绑定与解绑的监听器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 void valueBound(HttpSessionBindingEvent event) {
        System.out.println("person被绑定了");
    }
    @Override
    //解绑方法
    public void valueUnbound(HttpSessionBindingEvent event) {
        System.out.println("person被解绑了");
    }
}
protected void doGet(HttpServletRequest request, HttpServletResponse response)
        throws ServletException, IOException {
    HttpSession session = request.getSession();
    //将person对象绑到session中
    Person p = new Person();
    p.setId("100");
    p.setName("zhangsanfeng");  
    session.setAttribute("person", p);
    //将person对象从session中解绑
    session.removeAttribute("person");
}

钝化与活化的监听器HttpSessionActivationListener

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 void sessionWillPassivate(HttpSessionEvent se) {
        System.out.println("customer被钝化了");
    }
    @Override
    //活化
    public void sessionDidActivate(HttpSessionEvent se) {
        System.out.println("customer被活化了");
    }
}
protected void doGet(HttpServletRequest request, HttpServletResponse response)
        throws ServletException, IOException {
    HttpSession session = request.getSession();
    //将customer放到session中
    Customer customer =new Customer();
    customer.setId("200");
    customer.setName("lucy");
    session.setAttribute("customer", customer);
    System.out.println("customer被放到session域中了");
}
protected void doGet(HttpServletRequest request, HttpServletResponse response)
        throws ServletException, IOException {
    //从session域中获得customer
    HttpSession session = request.getSession();
    Customer customer = (Customer) session.getAttribute("customer");
    System.out.println(customer.getName());
}

可以通过配置文件 指定对象钝化时间 — 对象多长时间不用被钝化
在META-INF下创建一个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">
        <Store className="org.apache.catalina.session.FileStore" directory="it32" />
    </Manager>
</Context>

2、邮箱服务器

邮件的客户端:可以只安装在电脑上的也可以是网页形式的
邮件服务器:起到邮件的接受与推送的作用
邮件发送的协议:
协议:就是数据传输的约束
接受邮件的协议:POP3、IMAP
发送邮件的协议:SMTP

这里写图片描述

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");//发邮件的协议
        props.setProperty("mail.host", "localhost");//发送邮件的服务器地址
        props.setProperty("mail.smtp.auth", "true");//指定验证为true
        // 创建验证器
        Authenticator auth = new Authenticator() {
            public PasswordAuthentication getPasswordAuthentication() {
                return new PasswordAuthentication("tom", "12345");//发邮件的账号的验证
            }
        };
        Session session = Session.getInstance(props, auth);
        // 2.创建一个Message,它相当于是邮件内容
        Message message = new MimeMessage(session);
        message.setFrom(new InternetAddress("tom@it32.com")); // 设置发送者
        message.setRecipient(RecipientType.TO, new InternetAddress(email)); // 设置发送方式与接收者
        message.setSubject(subject);//邮件的主题
        message.setContent(emailMsg, "text/html;charset=utf-8");
        // 3.创建 Transport用于将邮件发送
        Transport.send(message);
    }
}

3、生日祝福

public class BirthdayListener implements ServletContextListener{

    @Override
    public void contextInitialized(ServletContextEvent sce) {
        //当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());
                //根据当前时间从数据查询今天过生日的人
                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) {
                    e.printStackTrace();
                } //08-18
                //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 (MessagingException e) {
                            e.printStackTrace();
                        }
                    }
                }       
            }
        }, new Date(), 1000*10);
        //实际开发中起始时间是一个固定的时间
        //实际开发中间隔时间是1天
    }
    @Override
    public void contextDestroyed(ServletContextEvent sce) { 
    }
}
  • 5
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值