过滤器

过滤器是一个java组件,可以拦截发送至某个servlet,jsp页面或静态页面的请求,可以在响应发送到客户之前进行拦截

工作原理:

过滤器类必须实现 Filter 接口,包含的方法如下:

void destroy()	 //销毁方法

void init(FilterConfig filterConfig) throws ServletException  //初始化方法

//主要的工作方法
void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws java.io.IOException, ServletException {
//这里写request的相关代码
chain.doFilter(request, resp);
//这里写response的相关代码
}

FilterChain chain 过滤器链

FilterChain 接口用于调用过滤器链中的下一个过滤器或调用过滤器结束后的资源

过滤器链如图:

过滤器生命周期的各个阶段:

实例化:Web容器在不是web应用程序的时候对所有过滤器进行实例化

	web容器回调它的无参构造方法

初始化:实例化完成之后,马上进行初始化工作

	web容器回调init方法

过滤:请求路径匹配过滤器的URL映射

	web容器回调 doFilter方法  --> 主要工作方法 

销毁:web容器在卸载web应用之前

	web容器回调 destroy方法

过滤器的实际应用:

1.对请求消息体中的数据设置统一的编码

2.阻止非法用户的请求

3.过滤非法数据

注意:

过滤器使用时需要在web.xml中配置,需要在“url-pattern”标签中指明过滤的对象 如“/*”过滤项目中所有文件,代码如下:

<filter>
<display-name>OurFilter</display-name>
<filter-name>OurFilter</filter-name>
<filter-class>nm.filter.OurFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>OurFilter</filter-name>
<url-pattern>/show.do</url-pattern>
</filter-mapping>

过滤器的简单实例应用

  我们都知道当我们百度搜索关键词的时候,搜索出的关键词会变成着重飘红。

接下来的通过过滤器,来简单的模仿百度的这种功能:

  • 用户请求的页面
package nm.filter;

import java.io.IOException;
import java.io.PrintWriter; import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse; public class OurServlet extends HttpServlet {
private static final long serialVersionUID = 1L; protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
request.setCharacterEncoding("utf-8");
response.setContentType("text/html;charset=utf-8"); PrintWriter out = response.getWriter();
//这里我们向页面输出4段带"我们"的句子
out.println("我们去玩吧<br/>");
out.println("你打不过我们<br/>");
out.println("我们一起吃饭去<br/>");
out.println("看我们的儿子<br/>");
} protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { doGet(request, response);
} }
  • 在这里我们写一个MyWriter类继承PrintWriter类 重写里面的write方法,将原来输出字符串的方法,改变成保存字符串的方法,另外再写一个获取字符串的方法,如下:
package nm.filter;

import java.io.PrintWriter;
import java.io.Writer; public class MyWriter extends PrintWriter{
private StringBuilder buffer; public MyWriter(Writer out) {
super(out);
buffer = new StringBuilder();
} @Override
public void write(String s) {
buffer.append(s);
} // 将write流中的内容全部转换为String
public String getContent(){
return buffer.toString();
} }
  • 这里我们写一个MyResponse类继承HttpServletResponseWrapper类(HttpServletResponse接口的实现类),来获取我们上面所写的MyWriter类:
package nm.filter;

import java.io.IOException;

import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpServletResponseWrapper; public class MyResponse extends HttpServletResponseWrapper{
private MyWriter myWriter; public MyResponse(HttpServletResponse response) {
super(response); } @Override
public MyWriter getWriter() throws IOException {
myWriter = new MyWriter(super.getWriter()); return myWriter;
} public MyWriter getMyWriter(){
return myWriter;
} }
  • 完成上面两个类之后,我们来写过滤器。通过过滤器我们将用户访问的页面拦截,将原本输出的信息用我们所写的类和方法代替,将里面的关键词“我们”进行修改,之后再利用原始的类和方法将信息输出:
package nm.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;
import javax.servlet.http.HttpServletResponse; public class OurFilter implements Filter { public void destroy() {
} public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
MyResponse resp = new MyResponse((HttpServletResponse)response);
chain.doFilter(request, resp); MyWriter writer = resp.getMyWriter();
if(writer!=null){
String content = writer.getContent();
String new_content = content.replace("我们","<span style='color:red'>我们</span>"); response.getWriter().write(new_content);
} } public void init(FilterConfig fConfig) throws ServletException { } }
最终的结果如图:


监听器

Listener 用于监听java web程序中的事件,比如 创建,修改,删除 Session,request,context等 ,并触发响应的事件

观察者模式:事件发生的时候会自动触发该事件对应 的Listener

Listener 主要对于 Session,request,context 进行监控

具体如下:

不同功能Listener需要实现的不同的Listener接口

一个Listener也可以实现多个接口,这样可以多种功能的监听器一起工作

8种监听器 分为三类

  1.监听Session,request,context的创建 和 销毁

       	HttpSessionListener  :监听Session的创建和销毁 

	        创建Session的时候执行 sessionCreate 方法 

	        当session的执行invalidate方法的时候,触发SessionDestroyed 方法

        ServletRequestListener  : 监听Request的创建和销毁

	        每次用户请求request都会执行requestInitialized方法

	        request处理完毕之后销毁之前执行 requestDestroyed 

	        注意:如果一个HTML页面中有多个图片,则每请求一次HTML页面可能会触发request事件

        ServletContextListener :监听Context的创建和销毁

	        服务器启动的时候 执行 contextInitialized 方法

	        服务器关闭或项目卸载 执行 contextDestroyed 方法

  2.监听对象属性编号 分别:

        Session属性变化	HttpSessionAttributeListener   

        Context属性变化	        ServletContextAttributeListener

        Request属性变化        ServletRequestAttributeListener

	        XXXXAdded();

	        XXXXReplaced();

	        XXXXRemoved() ;

	        说明:XXXX表示 Session context request

  3.监听Session内的对象

    	HttpSessionBindingListener

	        当对象被放到Session里执行  valueBound 方法

	        当对象从Session中移除执行  valueUnbound 方法

        HttpSessionActivationListener

	        服务器关闭的时候,会将Sessioin里的内容保存到硬盘上,这个过程叫 钝化

	        服务器重新启动的时候,会将session内容从硬盘中重新加载。

	        当Session中的对象被钝化的时候 sessionWillPassivate

	        当session中的对象被重新加载 执行sessionDidActivate

	        常用于session内的对象对session监听

         注意: 是Session内的对象,而不是Session本身,不需要web.xml配置

实现web.xml 的Listener 配置

 1. <listener> 标签 和 <listener-class>
2. <listener> 一般配置在 <servlet> 标签的前面   具体代码如下: <listener>
<listener-class>nm.listener.TestListener</listener-class>
</listener>

监听器简单的应用实例

  接下来,我们通过监听器来实现一个简单的统计用户在线人数的简单实例,具体代码如下:

  • 用户登录界面
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>用户登录界面</title>
</head>
<body>
<h1>用户登录界面</h1>
<form action="login.do" method="post">
用户名:<input type="text" name="uname" /><br/><br/><br/>
密 码:<input type="password" name="upwd" /><br/><br/><br/>
<input type="submit" value="提交"/>
<input type="reset" value="重置"/> </form>
</body>
</html>
  • 用户登录的简单处理程序,用户名为admin密码为123456
package servlet;

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 LoginServlet extends HttpServlet {
private static final long serialVersionUID = 1L; protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
request.setCharacterEncoding("utf-8");
response.setContentType("text/html;charset=utf-8");
String uname = request.getParameter("uname");
String upwd = request.getParameter("upwd");
HttpSession session = request.getSession();
//登陆成功后向页面输出当前在线人数
if(uname.equals("admin") && upwd.equals("123456")) {
response.getWriter().println("登陆成功~<br/>");
response.getWriter().println("当前在线人数 :"+(Integer)session.getServletContext().getAttribute("uerNumber")+"<br/>");
response.getWriter().println("<a href='DistoryServlet'>点击退出</a>");
}
} protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
doGet(request, response);
} }
  • 用户点击退出后的处理程序,这里的处理是删除会话session并重定向到登陆界面
package servlet;

import java.io.IOException;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse; public class DistoryServlet extends HttpServlet {
private static final long serialVersionUID = 1L; protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//清除session并重定向到登陆界面
request.getSession().invalidate();
response.sendRedirect("login.html");
} protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
doGet(request, response);
} }
  • 监听器处理程序,用一个变量来统计在线人数,每建立一个会话我们就让变量增加1,每销毁一个会话我们就让变量减少1
package listener;

import javax.servlet.http.HttpSessionEvent;
import javax.servlet.http.HttpSessionListener; public class LoginListener implements HttpSessionListener { int uerNumber = 0; //用来统计在线人数的变量 public void sessionCreated(HttpSessionEvent arg0) {
uerNumber++;
//将变量设为应用上下文的属性,便于在整个项目都可以访问到在线人数
arg0.getSession().getServletContext().setAttribute("uerNumber", uerNumber);
System.out.println("当前在线用户:"+uerNumber);
} public void sessionDestroyed(HttpSessionEvent arg0) {
uerNumber--;
//将变量设为应用上下文的属性,便于在整个项目都可以访问到在线人数
arg0.getSession().getServletContext().setAttribute("uerNumber", uerNumber);
System.out.println("当前在线用户:"+uerNumber);
} }

结果如图:


以上就是Servlet中的

过滤器(Filter)和监听器(Listener)

Servlet之过滤器(Filter)和监听器(Listener)的更多相关文章

  1. 过滤器Filter与监听器Listener

    过滤器Filter 过滤器也是一种servlet   它也可以对用户的请求进行处理  , 但是他所做的处理,只是一些轻量级的处理.Fileter就好像jsp页面与servlet之间的一道关卡,如果这个 ...

  2. 过滤器(filter),监听器(listener),与servlet的执行顺序

    创建: 加载顺序 监听器-->过滤器-->Servlet.项目启动后,容器会首先创建声明的各种监听器,为后继的各个事件监听做准备,然后创建过滤器,最后是Servlet.销毁的时候是反序进行 ...

  3. 二十五、过滤器Filter,监听器Listener,拦截器Interceptor的区别

    1.Servlet:运行在服务器上可以动态生成web页面.servlet的声明周期从被装入到web服务器内存,到服务器关闭结束.一般启动web服务器时会加载servelt的实例进行装入,然后初始化工作 ...

  4. Servlet过滤器Filter和监听器

    一.Servlet过滤器的概念: *********************************************************************************** ...

  5. JavaWeb学习篇之----Servlet过滤器Filter和监听器

    首先来看一下Servlet的过滤器内容: 一.Servlet过滤器的概念: ************************************************************** ...

  6. Servlet的过滤器Filter

    Servlet 编写过滤器 Servlet 过滤器可以动态地拦截请求和响应,以变换或使用包含在请求或响应中的信息. 可以将一个或多个 Servlet 过滤器附加到一个 Servlet 或一组 Serv ...

  7. 8-过滤器Filter和监听器Listener

    一.web监听器:监听特殊事件的发生1.监听实现步骤 a.写一个java类,实现特定的接口,重写相关方法 b.在web.xml中,牌配置 <listener> <listener-c ...

  8. Java Web(五) 监听器Listener

    监听器概述 在上一篇里介绍了过滤器Filter,而Listener是Servlet的另一个高级特性.Listener用于监听Java Web程序中的事件,例如创建,修改,删除Session,reque ...

  9. JavaWeb_初识过滤器Filter

    菜鸟教程 传送门 过滤器Filter::JavaWeb三大组件之一,它与Servlet很相似,过滤器是用来拦截请求的,而不是处理请求的 当用户请求某个Servlet时,会先执行部署在这个请求上的Fil ...

随机推荐

  1. 【转】深入浅出:Linux设备驱动之字符设备驱动

    深入浅出:Linux设备驱动之字符设备驱动 一.linux系统将设备分为3类:字符设备.块设备.网络设备.使用驱动程序: 字符设备:是指只能一个字节一个字节读写的设备,不能随机读取设备内存中的某一数据 ...

  2. 【ASP.NET MVC 学习笔记】- 11 Controller和Action(2)

    本文参考:http://www.cnblogs.com/willick/p/3331513.html 1.MVC一个请求的发出至action返回结果的流程图如下: 重点是Controller Fact ...

  3. MySql sql按时间分组

    select DATE_FORMAT(f.upload_time,'%Y%u') weeks,count(*),sum(p.download_times),sum(p.collection_times ...

  4. 【20171027早】alert(1) to win 第9,10,11,12题

    人在江湖,不服就干! 第9题: function escape(s) { function htmlEscape(s) { return s.replace(/./g, function(x) { r ...

  5. 将本地web服务映射到公网访问

    本文始发于我的个人博客,如需转载请注明出处. 为了更好的阅读体验,可以直接进去我的个人博客看. 项目部署 之前在学习前端的时候项目都只是在本地测试,永远的都是类似 http://localhost/x ...

  6. Android_简易的短信发送器

    这个随笔将介绍如何完成一个简单的第三方的短信发送器(不打开短信界面,调用android的api完成功能) 1.首先,我们来做布局 由于我这里写的是一个简易的,,短信发送,所以只是一个LinearLay ...

  7. Code Lock

    Code Lock Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 131072/65536 K (Java/Others) Total Su ...

  8. IdentityServer4 配置负载均衡

    如果使用 IdentityServer4 做授权服务的负载均衡,默认情况下是不可以的,比如有两个授权服务站点,一个资源服务绑定其中一个授权服务(Authority配置),如果通过另外一个授权服务获取a ...

  9. Linux系列教程(十七)——Linux权限管理之文件系统系统属性chattr权限和sudo命令

    上篇博客我们介绍了权限管理的ACL权限,通过设定 ACL 权限,我们为某个用户指定某个文件的特定权限.这篇博客我们将介绍权限管理中用的比较多的两个命令 chattr 和 sudo . 1.设定文件系统 ...

  10. 应用在安卓和ios端APP的证件识别

    移动端证件识别智能图文处理,是利用OCR识别技术,通过手机拍摄身份证图像或者从手机相册中加载证件图像,过滤身份证的背景底纹干扰,自动分析证件各文字进行字符切分.识别,最后将识别结果按姓名.地址.民族. ...