Servlet开发(2)-----过滤器和监听器
六、过滤器(过滤Servlet)
实现javax.servlet.Filter接口;
一般客户端发出请求后会交给Servlet;如果过滤器存在,则客户端发出的请求都是先交给过滤器,然后交给Servlet;
我们可以完成一些在执行Servlet之前必须要做的事,比如request.setCharacterEncoding("GBK");
必须实现以下方法:(因为Filter是一个接口,因此三个方法必须都实现)
1.public void init(FilterConfig config) throws ServletException{} // init方法在Web容器启动时就会调用;
2.public void doFilter(ServletRequest req,ServletResponse resp,FilterChain chain){} // doFilter的参数是ServletRequest和ServletResponse而不是Http的;
3.public void destroy(){}
注意:
1.FilterChain含有public void doFilter(ServletRequest req,ServletResponse resp){}
一般代码形式如下:
public void doFilter(ServletRequest req,ServletResponse resp,FilterChain chain){
chain.doFilter(req,resp); //执行Servlet操作;
}
则这个函数会调用两次,一次是执行chain.doFilter之前,一次是执行chain.doFilter()之后;
写完过滤器后,我们必须要限制过滤器调用的范围,即域名为多少时会调用过滤器,我们在web.xml 中进行配置;
<filter> <filter-name></filter-name> <filter-class></filter-class> </filter> <filter-mapping> <filter-name></filter-name> <url-pattern></url-pattern> <!--过滤器应用的范围,如果为/*,则如果域名设置形如/a 或/abc等都会调用过滤器--> <!-- 比如url-pattern中为 /* ,则对于根目录下的所有文件进行过滤--> </filter-mapping>
如果需要配置信息,则形式如下:
<filter> <filter-name></filter-name> <filter-class></filter-class> <init-param> <param-name></param-name> <param-value></param-value> </init-param> </filter> <filter-mapping> <filter-name></filter-name> <url-pattern></url-pattern> <!--过滤器应用的范围,如果为/*,则如果域名设置形如/a 或/abc等都会调用过滤器--> </filter-mapping>
七、监听器(监听Servlet)
监听器的作用类似于Swing中的监听器的作用,效果也差不多,即当某个事件发生时,就触发了某个设置好的监听器,但是这里触发的原因不同,这里是由于
(1)某个对象创建和销毁;
(2)设置和清除了内置对象的属性;
这里监听器能监听application、session、request对象。
写好监听器类后需要配置web.xml,形式如下:
<listener> <listener-class></listener-class> </listener>
1.application监听器:ServletContextListener
需要实现的方法:
(1)public void contextInitialized(ServletContextEvent e); //在application创建时就调用
(2)public void contextDestroyed(ServletContextEvent e); //当application销毁时调用
ServletContextEvent的getServletContext()方法可以取得application对象;
2.application属性监听器:ServletContextAttributeListener
需要实现的方法:
(1)public void attributeAdded(ServletContextAttributeEvent e); //当调用application.setAttribute()时调用
(2)public void attributeRemoved(ServletContextAttributeEvent e); //当调用applcaition.removeAttribute()时调用
(3)public void attributeReplaced(ServletContextAttributeEvent e); //当调用两次application.setAttribute()赋予相同属性时调用
ServletContextAttributeEvent 的方法有:
(1)getName(); 取得属性的名称;
(2)getValue(); 取得属性的值;(注意:返回的是Object,必须转型)
3.session监听器:HttpSessionListener
需要实现的方法:
(1)public void sessionCreated(HttpSessionEvent e); //当打开一个浏览器时,就会触发这个方法;
(2)public void sessionDestroyed(HttpSessionEvent e); //当调用session.invalidate();或超时时调用
HttpSessionEvent的方法有getSession()取得HttpSession内置对象;
销毁session:
(1)session.invalidate();
(2)超过超时时间,超时时间在web.xml中配置:
<session-config> <session-timeout>5</session-timeout> <!-- 5分钟 --> </session-config>
4.session属性监听器:HttpSessionAttributeListener
需要实现的方法:
(1)public void attributeAdded(HttpSessionBindingEvent e); //当调用session.setAttribute()时调用
(2)public void attributeRemoved(HttpSessionBindingEvent e); //当调用session.removeAttribute()时调用
(3)public void attributeReplaced(HttpSessionBindingEvent e); //当调用两次session.setAttribute()赋予相同属性时调用
HttpSessionBindingEvent 方法:
(1)getSession();
(2)getName();
(3)getValue();
5.session属性绑定监听器:HttpSessionBindingListener
需要实现的方法:
(1)public void valueBound(HttpSessionBindingEvent e);
(2)public void valueUnbound(HttpSessionBindingEvent e);
注意:这个监听器不用在web.xml中进行配置,而自动生效。
当实现这个接口的类被作为属性添加如内置对象时,就会触发valueBound;当删除这个属性时,则会触发valueUnbound;
比如
class A implements HttpSessionBindingListener{
.....
public void valueBound(HttpSessionBindingEvent e){}
public void valueUnbound(HttpSessionBindingEvent e){}
}
当调用
session.setAttribute("info",new A())时即添加A类对象时,则会触发valueBound方法,当调用session.removeAttribute("info")时触发valueUnbound方法;
6.request监听器:ServletRequestListener
需要实现的方法:
(1)public void requestInitialized(ServletRequestEvent e); //当请求一个网页时会调用
(2)public void requestDestroyed(ServletRequestEvent e); //当请求结束时会调用
ServletRequestEvent 方法:
(1)getServletContext();取得application对象;
(2)getServletRequest(); 取得request对象;
7.request属性监听器:ServletRequestAttributeListener
需要实现的方法:
(1)attributeAdded(ServletRequestAttributeEvent e); //当调用request.setAttribute()时调用
(2)attributeRemoved(ServletRequestAttributeEvent e); //当调用request.removeAttribute()时调用
(3)attributeReplaced(ServletRequestAttributeEvent e); //当调用两次request.setAttribute()赋予相同属性时调用
ServletRequestAttributeEvent 方法:
(1)getName();
(2)getValue();
监听器应用场景
ServletContextListener:在任何Servlet提供服务之前执行、在Servlet销毁时执行,用于提前初始化一些资源,比如数据库连接、销毁一些资源,比如数据库连接;
HttpSessionListener:多少个在线用户,即跟踪会话;
ServletContextAttributeListener:上下文中添加、删除、替换了属性;
ServletRequestListener:请求到来时记录日志;
ServletRequestAttributeListener:添加、删除、替换请求属性;
HttpSessionBindingListener:监听属性类的添加、删除;
HttpSessionAttributeListener:会话属性添加、删除、替换;
HttpSessionActivationListener:你有一个属性类,并且希望这个类的对象在绑定会话迁移到另一个JVM中;
HttpSessionAttributeListener&HttpSessionBindingListener
HttpSessionAttributeListener:当会话中添加、删除、替换属性时;
HttpSessionBindingListener:当属性本身作为会话的属性、从会话中删除时;
HttpSessionBindingListener方法:
(1)public void valueBound(HttpSessionBindingEvent event);
(2)public void valueUnbound(HttpSessionBindingEvent event);
注意:HttpSessionAttributeListener不需要在web.xml中注册;
补充:
1.HttpSessionActivationListener
会话迁移
在分布式Web应用中,一个web应用会在每个JVM中存在一个实例,因此对于WEBAPP1而言,在JVM1中有一个ServletContext实例,在JVM2中也会有一个ServletContext实例,而一个WEB应用的会话只有一个,因此如果客户向JVM1中的WEBAPP1发送请求,则会话需要迁移到JVM1中,反之,将会迁移到JVM2中;
HttpSessionActivationListener需要实现:
(1)public void sessionWillPassivate(HttpSessionEvent event);
(2)public void sessionDidActivate(HttpSessionEvent event);
2.会话超时
会话超时时间可以通过两种方式设定:
(1)在web.xml 中设定;
(2)在Servlet中设定;
在web.xml中设置:
<session-config>
<session-timeout>分钟值</session-timeout>
</session-config>
在Servlet中调用 session.setMaxInactiveInterval(秒钟值);
注意:如果会话超时时间设置为-1,则表示会话永远不会超时;