一、Listener监听器

Javaweb开发中的监听器,是用于监听web常见对象 HttpServletRequest HttpSession
ServletContext

监听它们的创建与销毁、属性变化 以及session绑定javaBean

1、监听机制

  • 事件  就是一个事情
  • 事件源  产生这个事情的源头
  • 监听器  用于监听指定的事件的对象
  • 注册监听 要想让监听器可以监听到事件产生,必须对其进行注册。

2、Javaweb开发中常见监听器

2.1、监听域对象的创建与销毁

  • 监听ServletContext创建与销毁  ServletContextListener
  • 监听HttpSession创建与销毁  HttpSessionListener
  • 监听HttpServletRequest创建与销毁  ServletRequestListener

2.2、监听域对象的属性变化

  • 监听ServletContext属性变化  ServletContextAttributeListener
  • 监听HttpSession属性变化  HttpSessionAttributeListener
  • 监听HttpServletRequest属性变化  ServletRequestAttributeListener

2.3、监听session绑定javaBean

它是用于监听javaBean对象是否绑定到了session域   HttpSessionBindingListener

它是用于监听javaBean对象的活化与钝化
 HttpSessionActivationListener

3、监听器的快速入门

关于创建一个监听器的步骤

  1. 创建一个类,实现指定的监听器接口
  2. 重写接口中的方法
  3. 在web.xml文件中对监听器进行注册。
  4. ServletContext对象的创建与销毁

3.1、关于域对象创建与销毁的演示

这个对象是在服务器启动时创建的,在服务器关闭时销毁的。

HttpSession对象的创建与销毁

HttpSession session=request.getSession();

Session销毁 的方法

  1. 默认超时  30分钟
  2. 关闭服务器
  3. invalidate()方法
  4. setMaxInactiveInterval(int interval) 可以设置超时时间

问题:直接访问一个jsp页面时,是否会创建session?

会创建,因为我们默认情况下是可以在jsp页面中直接使用session内置对象的。

HttpServletRequest创建与销毁

Request对象是发送请求服务器就会创建它,当响应产生时,request对象就会销毁。

3.2、演示了Request域对象中属性变化

在java的监听机制中,它的监听器中的方法都是有参数的,参数就是事件对象,而我们可以通过事件对象直接获取事件源。

3.3、**演示session绑定javaBean**

1、javaBean对象自动感知被绑定到session中.

HttpSessionBindingListener 这个接口是由javaBean实现的,并且不需要在web.xml文件中注册.

2、javabean对象可以活化或钝化到session中。

HttpSessionActivationListener如果javaBean实现了这个接口,那么当我们正常关闭服务器时,session中的javaBean对象就会被钝化到我们指定的文件中。

当下一次在启动服务器,因为我们已经将对象写入到文件中,这时就会自动将javaBean对象活化到session中。

我们还需要个context.xml文件来配置钝化时存储的文件

在meta-inf目录下创建一个context.xml文件

<Context>
<Manager className="org.apache.catalina.session.PersistentManager" maxIdleSwap="1">
<Store className="org.apache.catalina.session.FileStore" directory="it315"/>
</Manager>
</Context>

监听器Listener

案例-定时销毁session

1、怎样可以将每一个创建的session全都保存起来?

我们可以做一个HttpSessionListener,当session对象创建时,就将这个session对象装入到一个集合中.

将集合List<HttpSession>保存到ServletContext域中。

2、怎样可以判断session过期了?

在HttpSession中有一个方法public long
getLastAccessedTime()

它可以得到session对象最后使用的时间,可以使用invalidate方法销毁。

ps(我们上面的操作需要使用任务调度功能.)在java中有一个Timer定时器类

 

package com.timer;

import java.util.Date;
import java.util.Timer;
import java.util.TimerTask;

public class TestTimer {
    public static void main(String[] args) {
        Timer t = new Timer();

        t.schedule(new TimerTask() {

            @Override
            public void run() {
                System.out.println(new Date().toLocaleString());
            }
        }, 2000, 1000);
    }
}

关于三个域对象获取

如果在Servlet中要获取request,在方法上就有,request.getSession() 
getServletContext();

如果我们有request对象了,  request.getSession()
   request.getSession().getServletCotnext();

 

public class MySessionListener implements HttpSessionListener {

    public void sessionCreated(HttpSessionEvent arg0) {
        HttpSession session = arg0.getSession();
        //得到application中的list集合
        ServletContext application = session.getServletContext();
        //得到session对象,并放入list集合
        List<HttpSession> list =(List<HttpSession>) application.getAttribute("sessions");
        list.add(session);
        System.out.println("添加了"+session.getId());
    }
    public void sessionDestroyed(HttpSessionEvent arg0) {
        // TODO Auto-generated method stub
    }

}

程序在使用时,需要考虑并发问题,因为我们在web中,它一定是一个多线程的,那么我们的程序对集合进行了添加,还有移除操作。具体在MyServletContextListener的方法中如下

public class MyServletContextListener implements ServletContextListener {
    public void contextDestroyed(ServletContextEvent arg0) {   
    }
    public void contextInitialized(ServletContextEvent arg0) {
        //通過事件原對象得到事件源
            ServletContext application = arg0.getServletContext();
            //创建一个集合 存储所有session对象
            final List<HttpSession> list = Collections.synchronizedList(new ArrayList<HttpSession>());
            application.setAttribute("sessions", list);
            //创建一个计时器对象
            Timer t = new Timer();
            t.schedule(new TimerTask() {

                @Override
                public void run() {
                    System.out.println("开始扫描");
                    for (Iterator iterator = list.iterator(); iterator.hasNext();) {
                        HttpSession session = (HttpSession) iterator.next();
                        long l = System.currentTimeMillis() - session.getLastAccessedTime();
                        if(l > 5000){
                            System.out.println("session移除了"+session.getId());
                            session.invalidate();
                            iterator.remove();
                        }
                    }
                }
            }, 2000, 5000);    
    }
}

监听器概述

监听器就是一个Java类用来监听其他的JavaBean的变化.是一种观察者模式。
事件源:三大域对象 ServletContext,HttpSession,ServletRequest

二、Filter

 

创建步骤

  1. 事件相关监听器的接口
  2. 在web.xml中完成注册,Web3.0中支持注解(@WenListener)

二、Filter过滤器(重要)

Javaweb中的过滤器可以拦截所有访问web资源的请求或响应操作。

1.1、步骤:

  1. 创建一个类实现Filter接口
  2. 重写接口中方法  doFilter方法是真正过滤的。
  3. 在web.xml文件中配置

注意:在Filter的doFilter方法内如果没有执行chain.doFilter(request,response),那么资源是不会被访问到的。

监听的事件

1.2、FilterChain

FilterChain 是 servlet
容器为开发人员提供的对象,它提供了对某一资源的已过滤请求调用链的视图。过滤器使用
FilterChain
调用链中的下一个过滤器,如果调用的过滤器是链中的最后一个过滤器,则调用链末尾的资源。

问题:怎样可以形成一个Filter链?

  只要多个Filter对同一个资源进行拦截就可以形成Filter链

问题:怎样确定Filter的执行顺序?

  由<filter-mapping>来确定

第一类,对象初始化销毁监听

要实现的接口

{@link javax.servlet.http.HttpSessionListener},
{@link javax.servlet.ServletContextListener},
{@link javax.servlet.ServletRequestListener}

ServletContext对象何时创建和销毁

创建:服务器启动时候,服务器可以为每个WEB应用创建一个单独的ServletContext.
销毁:服务器关闭的时候,或者项目从服务器中移除.
用于监听对象创建的时候销毁的时候。常用于:初始化工作,加载配置文件,定时任务调度。

HttpSession对象何时创建和销毁

创建:服务器第一次调用getSession()方法的时候.

销毁
非正常关闭服务器(正常关闭序列化到硬盘)
session过期了(默认30分钟)
session.invalidate()
当以和请求访问的是jsp才会自动创建session对象。

ServletRequest对象何时创建和销毁

创建:客户端向服务器发送请求的时候.
销毁:服务器为这次请求作出了响应时候.

1.3、Filter生命周期

Servlet生命周期:

实例化 –》 初始化 –》 服务 –》 销毁

  • 1 当服务器启动,会创建Filter对象,并调用init方法,只调用一次.
  • 2
    当访问资源时,路径与Filter的拦截路径匹配,会执行Filter中的doFilter方法,这个方法是真正拦截操作的方法.
  • 3 当服务器关闭时,会调用Filter的destroy方法来进行销毁操作.
第二类,对象的属性监听

要是实现的接口

{@link javax.servlet.http.HttpSessionAttributeListener},
{@link javax.servlet.ServletContextAttributeListener},
{@link javax.servlet.ServletRequestAttributeListener},

属性监听 setAttribute addAttribute removeAttribute.

1.4、FilterConfig

在Filter的init方法上有一个参数,类型就是FilterConfig.

FilterConfig它是Filter的配置对象,它可以完成下列功能

  1. 获取Filtr名称
  2. 获取Filter初始化参数
  3. 获取ServletContext对象。

图片 1

问题:怎样在Filter中获取一个FIlterConfig对象?

 1 package com.itheima.filter;
 2 
 3 import java.io.IOException;
 4 
 5 import javax.servlet.Filter;
 6 import javax.servlet.FilterChain;
 7 import javax.servlet.FilterConfig;
 8 import javax.servlet.ServletException;
 9 import javax.servlet.ServletRequest;
10 import javax.servlet.ServletResponse;
11 
12 public class MyFilterConfigTest implements Filter{
13 
14     private FilterConfig filterConfig;
15 
16     public void init(FilterConfig filterConfig) throws ServletException {
17         this.filterConfig = filterConfig;
18     }
19 
20     public void doFilter(ServletRequest request, ServletResponse response,
21             FilterChain chain) throws IOException, ServletException {
22         //通过FilterConfig对象获取到配置文件中的初始化信息
23         String encoding = filterConfig.getInitParameter("encoding");
24         System.out.println(encoding);
25         request.setCharacterEncoding(encoding);
26         //放行
27         chain.doFilter(request, response);
28     }
29 
30     public void destroy() {
31         // TODO Auto-generated method stub
32     }
33 }

如下 web.xml配置
<filter>
    <filter-name>MyFilterConfigTest</filter-name>
    <filter-class>com.itheima.filter.MyFilterConfigTest</filter-class>
    <init-param>
        <param-name>encoding</param-name>
        <param-value>UTF-8</param-value>
    </init-param>
</filter>

<filter-mapping>
    <filter-name>MyFilterConfigTest</filter-name>
    <servlet-name>ServletDemo2</servlet-name>
</filter-mapping>

session的序列化反序列化,钝化和活化

session序列化

当服务器关闭后,tomcat的work的项目目录下会多出一个SESSIONS.ser(扩展名,序列化),
当服务器重启后,seesion对象又被读回到session域中

所以当客户端没有关闭时,即使服务器重启,也可以通过sessionid找回session对象。

去掉session的序列化功能

在tomcat的conf/context.xml下加入一个元素

<Manager pathname="">

session的钝化和活化

在session没有消失期间,过多session大量占用内存时,会将暂时不活动的session存入硬盘中,当用户在session有效期访问时再加载回来,这个过程就是session的钝化和活化

<!-- 配置到context下表示全局激活seesion的钝化和活化功能,这个过程就是序列化到硬盘的过程-->
<!-- 也可以在项目下/meta-info创建一个context.xml,并添加属性 -->
<!-- maxIdleSwap最大不活动时间,单位分钟 如果session不使用就会序列化到硬盘.-->
<Manager className="org.apache.catalina.session.PersistentManager" maxIdleSwap="1"/>
    <!-- 配置存储目录,序列化到硬盘的文件存放的位置. ,存储方式sessionid.session-->
    <Store className="org.apache.catalina.session.FileStore" driectory="mysession">
</Manager>

1.5、Filter配置

基本配置

<filter>
<filter-name>filter名称</filter-name>
<filter-class>Filter类的包名.类名</filter-class>
</filter>
<filter-mapping>
<filter-name>filter名称</filter-name>
<url-pattern>路径</url-pattern>
</filter-mapping>

 

关于其它配置

第三类,JavaBean感知监听

要实现的接口

HttpSessionBindingListener
HttpSessionActivationListener

监听HttpSession对象中的JavaBean的状态的改变.(绑定,解除绑定,钝化和活化)
这两个监听器比较特殊,它是由javaBean来实现的,并且不需要在web.xml文件中注册监听.
javaBean必须是序列化的.


为JavaBean实现HttpSessionBindingListener

功能:使javaBean对象在被绑定到会话或从会话中取消对它的绑定时得到通知

//该接口javaBean要实现下面的方法
//绑定时调用
@Override
public void valueBound(HttpSessionBindingEvent event) {
}

//解绑时调用
@Override
public void valueUnbound(HttpSessionBindingEvent event) {
}

javaBean序列化HttpSessionActivationListener

功能:绑定到会话的对象可以侦听通知它们会话将被钝化和会话将被激活的容器事件,让Bean对象和session一同序列化。

//实现下列方法,监听bean对象和session钝化和活化的情况。
@Override
public void sessionWillPassivate(HttpSessionEvent se) {
}

@Override
public void sessionDidActivate(HttpSessionEvent se) {
}

1.<url-pattern>

完全匹配   以”/demo1”开始,不包含通配符*

目录匹配   以”/”开始  以*结束

扩展名匹配  *.xxx  不能写成/*.xxx

过滤器Filter

他会在一组资源(静态,动态资源)前面执行,他可以让请求得到目标资源,也可以不让请求达到,过滤器有拦截请求的功能

2.<servlet-name>

它是对指定的servlet名称的servlet进行拦截的。

生命周期

当服务器启动,会创建Filter对象,并调用init方法,只调用一次.
当访问资源时,路径与Filter的拦截路径匹配,会执行Filter中的doFilter方法,这个方法是真正拦截操作的方法.
当服务器关闭时,会调用Filter的destroy方法来进行销毁操作.

3.<dispatcher>

可以取的值有  REQUEST  FORWARD  ERROR  INCLUDE    根据跳转方式拦截

它的作用是:当以什么方式去访问web资源时,进行拦截操作.

1.REQUEST
当是从浏览器直接访问资源,或是重定向到某个资源时进行拦截方式配置的
它也是默认值

2.FORWARD 它描述的是请求转发的拦截方式配置

3.ERROR
如果目标资源是通过声明式异常处理机制调用时,那么该过滤器将被调用。除此之外,过滤器不会被调用。

4.INCLUDE
如果目标资源是通过RequestDispatcher的include()方法访问时,那么该过滤器将被调用。除此之外,该过滤器不会被调用

配置办法

配置方式web.xml中添加filter标签或者web3.0中添加@WebFilter注解。

<!-- 关于servlet-name配置 -->
<!-- 在Filter中它的url-pattern配置项上有一个标签,它用于设置当前Filter拦截哪一个servlet。 -->
<servlet-name>

<!-- 请求方式过滤配置办法 -->
<dispatcher>REQUEST|FORWARD|INCLUDE|ERROR</dispatcher>
<!-- 可以配置多个该种标签 -->
<!-- REQUEST 当是从浏览器直接访问资源,或是重定向到某个资源时进行拦截方式配置的 它也是默认值 -->
<!-- FORWARD 它描述的是请求转发的拦截方式配置 -->
<!-- ERROR 如果目标资源是通过声明式异常处理机制调用时,那么该过滤器将被调用。除此之外,过滤器不会被调用。 -->
<!-- INCLUDE 如果目标资源是通过RequestDispatcher的include()方法访问时,那么该过滤器将被调用。除此之外,该过滤器不会被调用 -->

 三、自动登录  (PS bean的属性名和数据库里的字段名千万要一样哦 不然会emmm 很惨)

  1. 当用户登陆成功后,判断是否勾选了自动登陆,如果勾选了,就将用户名与密码持久化存储到cookie
  2. 做一个Filter,对需要自动登陆的资源进行拦截

首写要在登录Servlet中登录成功时把登录的信息保存在cookie里,并设置存活时间

if(null != user){
            String autologin = request.getParameter("autologin");

            Cookie cookie = new Cookie("user", user.getUsername()+"&"+user.getPassword());
            cookie.setPath("/");
            if(autologin != null){// on / null
                cookie.setMaxAge(60*60*24*7);
            }else{
                cookie.setMaxAge(0);
            }
            response.addCookie(cookie);
            request.getSession().setAttribute("user", user);
            request.getRequestDispatcher("/home.jsp").forward(request, response);
        }

 然后设置好filter类 记住要在web.xml里面配置  如下 为 doFilter代码

public void doFilter(ServletRequest arg0, ServletResponse arg1,
            FilterChain arg2) throws IOException, ServletException {
        //1、转换对象
        HttpServletRequest req = (HttpServletRequest) arg0;
        //2、处理业务
        Cookie[] cookies = req.getCookies();
        String username = "";
        String password = "";
        for (int i = 0;cookies!=null && i < cookies.length; i++) {
            if("user".equals(cookies[i].getName())){
                String value = cookies[i].getValue();
                String[] values = value.split("&");
                username = values[0];
                password = values[1];
            }
        }
        UserService us = new UserService();
        User user = us.findUser(username, password);
        if(user != null){
            req.getSession().setAttribute("user", user);
        }
        //3、放行
        arg2.doFilter(arg0, arg1);
    }

多个过滤器的执行顺序

多连接器的情况下,是先进后出,拦截的顺序和xml中的filter-mapping中的位置顺序有关。存在一个filterchain链,执行过滤器到最后一和过滤器。

国际化

把和语言相关的所有字符串都写成变量。

关键类:Locale locale = Locale.枚举值
来加载相关的资源文件,来进行中英文转化

相关文章