JavaWeb(过滤器&监听器)

1.Filter过滤器

  1. 概述
    过滤器是JavaWeb三大组件之一(servlet、Listener、Filter),过滤器实际上就是对web资源(HTML、CSS、Servlet、JSP)进行拦截,做一些处理后再交给下一个过滤器或servlet处理,通常都是用来拦截request进行处理的,也可以对返回的response进行拦截处理
    在这里插入图片描述

  2. 过滤器使用场景
    ① 自动登录
    ② 统一设置编码格式
    ③ 访问权限控制
    ④ 敏感字符过滤等

  3. Filter快速入门
    (1)编写一个Filter,定义类实现接口Filter,实现接口中的抽象方法

接口:

public class FilterDemo1 implements Filter{}

方法:

/**
  * 过滤器对象被销毁之前执行的方法 
  * 	服务器关闭或重载时,过滤器对象销毁
  */
public void destroy() {
	System.out.println("----过滤器被销毁----");
}
/**
  * 对请求进行过滤的方法 
  * 	ServletRequest 请求对象 
  * 	ServletResponse 响应对象 
  * 	FilterChain 过滤链对象
  * 		作用:将通过此过滤器的请求,传递给下一个过滤器或目标Servlet
*/
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain)
			throws IOException, ServletException {
	System.out.println("----过滤器执行过滤方法----");
	System.out.println("----DemoFilter-doFilter执行之前----");
	// 对请求放行的方法
	chain.doFilter(request, response);
	System.out.println("----DemoFilter-doFilter执行之后----");
}
/**
  * 过滤器对象创建之后执行的方法,对过滤器进行初始化 
  * 	其中FilterConfig参数为当前过滤器的配置,获取过滤器的初始化参数
  * 	当服务器启动时,过滤器对象被创建
*/
public void init(FilterConfig fConfig) throws ServletException {
	System.out.println("----过滤器初始化----");
}

(2)在web.xml文件中配置访问过滤路径(和Servlet的配置有些相似)

<filter>
  	<filter-name>FilterDemo</filter-name>
  	<filter-class>com.zking.filter.FilterDemo</filter-class>
  </filter>
  <filter-mapping>
  	<filter-name>FilterDemo</filter-name>
  	<url-pattern>/*</url-pattern>
</filter-mapping>
  1. 常用配置项 urlPatterns
    ① 以指定资源匹配。例如"/index.jsp"
    ② 以目录匹配。例如"/servlet/"
    ③ 以后缀名匹配,例如"
    .jsp"
    ④ 通配符,拦截所有web资源。“/*”

    过滤路径:

  • 具体路径:某个Servlet的路径
    例如:/hello
  • 路径后缀统配:*.后缀名
    例如: *.do *.html *.action
  • 全路径统配: /*
    过滤所有的请求,包括请求Servlet,jsp,以及请求静态资源
  1. Filter生命周期
    (1)构造器:Filter的构造器在服务器启动时调用。 构造器只会调用一次,说明Filter也是单例多线程的。
    (2)init():在构造器被调用后,紧接着被调用。作用:用来初始化Filter。
    (3)doFilter():每一次拦截请求时都会调用。
    (4)destroy 方法在项目停止时调用,用来在对象被销毁前做一些收尾工作。

  2. 多个Filter的执行顺序
    (1)在web.xml中,filter执行顺序跟的顺序有关,先声明的先执行
    (2)使用注解配置的话,filter的执行顺序跟名称的字母顺序有关,例如AFilter会比BFilter先执行
    (3)如果既有在web.xml中声明的Filter,也有通过注解配置的Filter,那么会优先执行web.xml中配置的Filter

  3. 过滤器在WEB应用中实例

实现接口重写方法:

package com.zking.filter;
 
import java.io.IOException;
 
import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
 
/**
 * 字符编码过滤器 
 * 	对所有的请求和响应对象设置字符集 
 * 	用来提高代码复用性和解决硬编码问题。
 *
 */
public class EncodingFilter implements Filter {
 
	private String encoding;
 
	@Override
	public void destroy() {
 
	}
 
	@Override
	public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain)
			throws IOException, ServletException {
		System.out.println("------字符集过滤器开始过滤------");
		// 对请求对象设置字符集
		servletRequest.setCharacterEncoding(encoding);
		// 对相应对象设置字符集
		servletResponse.setCharacterEncoding(encoding);
		servletResponse.setContentType("text/html;charset=" + encoding);
		// 放行
		filterChain.doFilter(servletRequest, servletResponse);
 
	}
 
	@Override
	public void init(FilterConfig filterConfig) throws ServletException {
		String encoding = filterConfig.getInitParameter("encoding");
		this.encoding = encoding;
	}
 
}

配置xml:

<filter>
	<filter-name>EncodingFilter</filter-name>
  	<filter-class>com.zking.filter.EncodingFilter</filter-class>
  	<!-- 把过滤器要用到的字符集定义在初始化参数里面 -->
  	<init-param>
  		<param-name>encoding</param-name>
  		<param-value>utf-8</param-value>
  	</init-param>
</filter>
<filter-mapping>
	<filter-name>EncodingFilter</filter-name>
  	<url-pattern>/*</url-pattern>
</filter-mapping>

2.Listener监听器

1.概述

  • 监听器就是一个专门用于:对其它对象产生的特定事件,或状态改变后进行监听和相应处理的对象
  • 监听器其实就是一个实现特定接口的普通 Java程序,这个程序专门用于监听另一个 Java对象的方法调用或属性改变。
  • 当被监听对象发生上述事件后,监听器某个方法立即被执行。
  1. 使用场景:
    (1)统计在线人数和在线用户
    (2)系统启动时加载初始化信息
    (3)统计网站访问量
    (4)跟Spring结合,做相关操作

  2. 3种常用监听器
    (1)application监听器 servlet—ServletContext

实现:ServletContextListener (常用)
重写方法:

  //容器启动时调用
  public void contextInitialized(ServletContextEvent event){
    
  }
 
  //容器消毁时调用
  public void contextDestroyed(ServletContextEvent event){
    
  }

ServletContextEvent事件方法:
event.getServletContext().getContextPath();//取得ServletContext对象,即上下文

(2)Session监听器
实现:HttpSessionListener (偶尔用)
重写:

 //session创建时调用
  public void sessionCreated(HttpSessionEvent event){
 
  }
 
  //session销毁时调用
  public void sessionDestroyed(HttpSessionEvent event){
 
  }

HttpSessionEvent事件方法:
event.getSession().getId(); //得到session的ID

实现:HttpSessionAttributeListener (不用,性能差)
重写:

  //增加属性时触发
  public void attributeAdded(HttpSessionBindingEvent event){
    
  }
 
  //删除属性时触发
  public void attributeRemoved(HttpSessionBindingEvent event){
    
  }
 
  //替换属性时触发
  public void attributeReplaced(HttpSessionBindingEvent event){
    
  }

HttpSessionBindingEvent事件方法:
event.getSession() //取得session
event.getName() //取得属性的名称
event.getValue() //取得属性的内容

(3)request监听器
实现:ServletRequestListener (不用,性能差)
重写:

  //请求开始时调用
  public requestInitialized(ServletRequestEvent event){
 
  }
 
  //请求结束时调用
  public requestDestroyed(ServletRequestEvent event){
 
  }

ServletRequestEvent事件方法:
event.getServletRequest().getRemoteAddr(); //得到IP地址
event.getServletContext().getContextPath(); //得到当前路径

配置xml:

 
<listener>
   <listener-class>com.listener.Application</listener-class>
</listener>

案例统计在线人数和在线用户:

① 编写监听事件

// 首先我们需要实现 Servlet规定的监听器接口
public class OnlineCountListener implements HttpSessionListener {
    // 实现该接口后会必须重写下面这两个方法
    @Override
    public void sessionCreated(HttpSessionEvent se) {
        // 该方法是会在 Session创建时被调用,也就是 Session创建的监听事件
 
        // 拿到上下文对象
        ServletContext context = se.getSession().getServletContext();
        Integer onlineCount = (Integer) context.getAttribute("onlineCount");
        // 在触发 Session创建监听事件时,如果 onlineCount变量为 0我们将其设置为 1,表示第一个用户在线
        if (onlineCount==null){
            onlineCount = new Integer(1);
            // 如果不为 0表示之前有用户在线,我们将在线人数 +1
        }else {
            int count = onlineCount.intValue();
            onlineCount = new Integer(count+1);
        }
        // 打印输出 方便测试,可以去掉
        System.out.println(onlineCount);
        // 将在线人数的变量赋值添加到上下文对象中,方便前端取值
        context.setAttribute("onlineCount",onlineCount);
    }
 
    @Override
    public void sessionDestroyed(HttpSessionEvent se) {
        // 这个方法则相反,会在Session被销毁时调用
 
        // 销毁部分则逻辑相反
        ServletContext context = se.getSession().getServletContext();
        Integer onlineCount = (Integer) context.getAttribute("onlineCount");
        if (onlineCount==null){
            onlineCount = new Integer(0);
        }else {
            int count = onlineCount.intValue();
            onlineCount = new Integer(count-1);
        }
        context.setAttribute("onlineCount",onlineCount);
    }
}

② 配置xml

<listener>
    <!--过滤器的注册相对比较简单-->
    <listener-class>com.molu.listener.OnlineCountListener</listener-class>
</listener>

③ 布局模拟

<body>
<div>
    <h2>当前网站在线人数为:
        <span style="background-color: aquamarine">
            ${applicationScope.get("onlineCount")}
        </span>
    </h2>
</div>
</body>
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值