监听器 Listener
监听分为三中
ServletContextListener
HttpSessionListener
ServletRequestListener(了解即可)
监听器能够针对创建和销毁定义不同的行为
今天我们来看看ServletContextListenerd与HttpSessionLisrener
我们选择用一个表单来做案例:
表单:
<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
<!DOCTYPE html>
<html>
<head>
<%@taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c"%>
<meta charset="UTF-8">
<title>Insert title here</title>
</head>
<body>
<h1>当前人数:${onLineCount }</h1>
<h3><a href="exit.do">退出</a></h3>
<h3><a href="admin_a.jsp">管理员A</a></h3>
<h3><a href="admin_b.jsp">管理员B</a></h3>
<h3><a href="tourists.jsp">游客界面</a></h3>
<c:if test="${uname==null}">
<form action="login.do">
用户名:<input type="text"name="uname">
密码:<input type="password" name="upwd">
<button>提交</button>
</form>
</c:if>
<c:if test='${uname==""}'>
<h3>欢迎登陆</h3>
</c:if>
</body>
</html>
我们先来实现记录网站在线会话人数
因为实现这个需要继承两个实现两个接口
所以我们的思路就很明确了:
思路:
1.先继承并实现两个接口自带的方法
2.定义一个全局变量(ServletContext application)
3.根据代码上往下走
application将在contextInitialized(ServletContextEvent sce)方法中被初始化(初始人数为0当回话被进入的时候加一)application失效的时候减一就能实现在先记录网站人数
代码:
package com.zking.Listener;
import javax.servlet.ServletContext;
import javax.servlet.ServletContextEvent;
import javax.servlet.ServletContextListener;
import javax.servlet.annotation.WebListener;
import javax.servlet.http.HttpSessionEvent;
import javax.servlet.http.HttpSessionListener;
import org.apache.catalina.SessionEvent;
import org.apache.catalina.SessionListener;
@SuppressWarnings("all")
@WebListener//配置监听器
public class OnLineListener implements ServletContextListener,HttpSessionListener{
ServletContext application;//全局对象
@Override
public void contextDestroyed(ServletContextEvent sce) {
// application被销毁了
System.out.println("服务器销毁了");
}
@Override
public void contextInitialized(ServletContextEvent sce) {
//application被创建了
System.out.println("服务器启动了");
//将程序的上下文赋值给全局变量
application = sce.getServletContext();
//项目启动的时候 把人数设置为0
application.setAttribute("onLineCount", 0);
}
@Override
public void sessionCreated(HttpSessionEvent se) {
//只要该项目的页面被访问了
//获取application中存放的人数
Integer count=(Integer)application.getAttribute("onLineCount");
//设置人数 人数+1
application.setAttribute("onLineCount", ++count);
System.out.println("有人进来了,人数:"+count);
}
@Override
public void sessionDestroyed(HttpSessionEvent se) {
//1.存活时间 ttl 到期了
//2.手动销毁 req.getsession().invalidate();
//获取application中存放的人数
Integer count=(Integer)application.getAttribute("onLineCount");
//设置人数 人数+1
application.setAttribute("onLineCount", --count);
System.out.println("有人出去了,人数:"+count);
}
}
这就是监听器最基本的使用了
过滤器
基本介绍: 过滤器,顾名思义就是对事物进行过滤的,在Web中的过滤器,当然就是对请求进行过滤,我们使用过滤器,就可以对请求进行拦截,然后做相应的处理,实现许多特殊功能。如登录控制,权限管理,过滤敏感词汇等.
原理: 当我们使用过滤器时,过滤器会对游览器的请求进行过滤,过滤器可以动态的分为3个部分,1.放行之前的代码,2.放行,3.放行后的代码,这3个部分分别会发挥不同作用。
- 第一部分代码会对游览器请求进行第一次过滤,然后继续执行
- 第二部分代码就是将游览器请求放行,如果还有过滤器,那么就继续交给下一个过滤器
- 第三部分代码就是对返回的Web资源再次进行过滤处理
怎么使用: 我们使用过滤器肯定要导入相应的jar包才行,Filter就在servlet-api.jar中,我们将该jar包放到WEB-INF下的lib目录下面,然后加入项目。
然后继承Filter即可
import javax.servlet.*;
import javax.servlet.annotation.WebFilter;
import java.io.IOException;
@WebFilter("/*")
public class MyFilter implements Filter {
@Override
public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
System.out.println("对request进行过滤");
//下面这行代码就是放行
filterChain.doFilter(servletRequest,servletResponse);
System.out.println("对response进行过滤");
}
}
我简单介绍下上面的代码,WebFilter("/*")表示对所有请求进行过滤,而在doFilter中的放行代码,也就是filterChain.doFilter(servletRequest,servletResponse);这行代码就是对拦截进行放行,细节我们后面讲,现在先怎么理解就行。
启动服务器,然后我们在游览器中输入http://localhost:8080/filter/abc,注意,filter是我们自己配置的web工程路径,后面的abc随便输入的。我们下面来查看游览器后控制台输出。
我们在上面创建的类中写入以下代码,并且加一个WebFIlter注解
配置拦截:
package com.zking.Filter;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.annotation.WebFilter;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
/**
* 权限过滤器【鉴权】
* @author zjjt
*
*/
@WebFilter("/*") //设置过滤规则
public class RoleFilter implements Filter{
//放我不需要过滤的路径
List<String> paths=new ArrayList<String>();
//将路径放到列表中
{
paths.add("/index.jsp");
paths.add("/tourists.jsp");
paths.add("/login.do");
paths.add("/exit.do");
}
@Override
public void doFilter(ServletRequest req, ServletResponse resp, FilterChain chain)
throws IOException, ServletException {
//过滤器的所有操作全部在这里完成
HttpServletRequest request=(HttpServletRequest)req;
HttpServletResponse response=(HttpServletResponse)resp;
//获取当前请求的路径
String path=request.getServletPath();
//判断你当前访问的路径是否需要过滤
boolean f=false;
for(String p :paths) {
if(p.equals(path)) {
f=true;
break;
}
}
if(f) {//当你的访问路径在列表中 我是不需要过滤的
//让过滤器放行
chain.doFilter(req, resp);
return;//终止代码运行
}
//uname是在登陆之后被放到session里面去
Object uname = request.getSession().getAttribute("uname");
if(uname==null) {//没有登陆
//回去首页
response.sendRedirect("index.jsp");
return;
}
//让过滤器放行
chain.doFilter(req, resp);
}
@Override
public void init(FilterConfig arg0) throws ServletException {
// TODO Auto-generated method stub
}
@Override
public void destroy() {
//过滤器的所有操作都在这里完成
}
}