Servlet-jsp从入门到精通(三)
监听器
Servlet API中定义了8个监听器接口,可以用于监听ServletContext,HttpSession,ServletRequest对象的生命周期事件。
-
ServletContextListener 监听Servlet上下文对象初始化或者被销毁
-
ServletContextAttributeListener 监听Servlet上下文中的属性列表的变化
-
HttpSessionListener 监听Session生命周期
-
HttpSessionActionListener 监听session被钝化或者激活
-
HttpSessionAttributeListener 监听Session属性列表发生的变化
-
HttpSessionBindingListener 监听Session中是否有对象绑定或者删除,该对象要实现这个接口
-
ServletRequestListener 监听ServletRequest对象生命周期
-
ServletRequestAttributeListener 监听ServletRequest属性列表发生的变化
详细解释:
-
ServletContextListener 监听ServletContext的启动或者销毁
-
contextInitialized(ServletContextEvent sce)
当web应用程序初始化进程正开始时,web容器调用这个方法,该方法将在所有的过滤器和Servlet初始化之前被调用。contextDestroyed(ServletContextEvent sce)当Servlet上下文将要关闭时,Web容器调用这个方法,该方法在所有Servlet和和过滤器销毁之后被调用。课堂例子: 在Web应用程序启动时初始化DataSource对象,然后将其方法ServletContext中 数据库支持更换
-
HttpSessionBindingListener,(无序配置)
如果一个对象实现了HttpSessionBindingListener接口,当这个对象被绑定的Session或者从Session中被删除时,Servlet容器就会通知这个对象。 -
valueBound(HttpSessionBindingEvent event)
当对象正在被绑定到Session中,Servlet容器调用这个方法通知该对象 -
valueUnBound(HttpSessionBindingEvent event)
当从Session中删除对象时,Servlet容器调用这个方法通知该对象应用实例: 购物车对象实现该接口,当顾客选购商品时,web应用程序创建购物车对象保存到Session中。当Session超时时,Servlet容器会通知购物车,它要从Session中被删除了。购物车对象在得到通知后,可以把顾客选购的商品信息保存到数据库中。
过滤器
在源数据和目的数据之间起过滤作用的中间组件
当web容器接收到一个对资源的请求时,它将判断是否有过滤器与这个资源相关,如果有,那么容器将请求交给过滤器进行处理。在过滤器中,可以改变请求的内容或者重新设置请求的报头信息,然后再将请求发送给目标资源。当目标资源对请求作出响应时,容器同样会将响应转发给过滤器。然后发送到客户端。
Filter接口
- init(FilterConfig filterConfig)
初始化过滤器。filterconfig用来获取ServletContext,初始化参数。 - doFilter(ServletRequest request,ServletResponse response,FilterChain chain);
实现过滤功能。可以调用chain.doFilter(request,response)方法将请求传递给下一个过滤器或者目标资源,也可以直接向客户端返回响应信息。 - destory()
结束过滤器的声明周期。
FilterConfig接口
- getInitParameter(String name)
返回名为name的初始化参数的值 - getInitParameterNames()
返回所有初始化参数的名字的枚举集合。 - getServletContext()
返回Servlet上下文对象的引用
FilterChain接口
- doFilter(ServletRequest request,ServletResponse response)
调用该方法将使过滤器链中的下一个过滤器被调用,如果该方法时过滤器链中最后一个过滤器,那么目标资源被调用。
过滤器的配置
<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>
<!--指定过滤器对应的请求方式,默认request-->
<dispatcher>REQUEST,INCLUDE,FORWARD,ERROR</dispatcher>
</filter-mapping>
dispatcher:
- REQUEST 当用户直接访问页面时,web容器将会调用过滤器,但是当通过include,forward()方法调用时,该过滤器不会被调用
- INCLUDE 目标资源通过include()方法访问该过滤器才会被调用
- FORWARD 目标资源通过forward()方法访问该过滤器才会被调用
- ERROR 目标资源是通过声明式异常处理机制调用时,该过滤器才会被调用
开发线程安全的Servlet
在大型web应用中,Servlet容器可能同时会接收到多个客户端的上千个请求,可以采用多线程并发地处理每个请求。
多线程的Servlet模型
Servlet容器维护了一个线程池服务请求,线程池实际上是等待执行代码的一组线程,这些线程叫做工作线程,Servlet容器使用一个调度者线程来管理工作者线程,当容器接收到一个访问Servlet的请求,调度者线程从线程池中选取一个工作者线程,将请求传递给该线程,然后由这个线程执行Servlet的service()方法。由于Servlet是单例的,所有Servlet的service方法将在多个线程中并发地执行。Servlet容器采用了单实例多线程的方法,最大限度地减少了Servlet实例的开销,显著提升了对请求的响应时间。
ServletContext 是线程不安全的。
我们应当合理设计系统,在Servlet上下文中只保存很少修改的数据,而对于其他经常需要修改的数据,应该采用另外的方式在多个Servlet中共享HttpSession 是线程不安全的。HttpSession 只能在处理属于同一个Session的请求的线程中被访问但是用户可以打开多个同属于一个进程的浏览器窗口,这些窗口中的访问请求属于同一个Session为了同时处理多个这样的请求Servlet容器会创建多个线程,就会出现多线程并发访问Session中的数据,为解决这个问题,可以对Session的访问进行同步
synchronized(session){
}
ServletRequest线程安全的。Servlet容器对它所接受的每个请求,都创建一个新的ServletRequest对象,所以ServletRequest对象旨在一个线程中被访问。
最佳实践:
- 尽可能地在Servlet中只使用本地变量
- 应该只使用只读的实例变量和静态变量
- 不要在Servlet中创建自己的线程
- 修改共享对象时,一定要使用同步,尽可能地缩小同步代码的范围,不要直接在service方法或者doXXX方法中进行同步,以免影响性能
- 如果在多个不同的Servlet中,要对外部对象(例如 文件)进行修改操作,一定要加锁.
- 在多线程的环境中使用集合类对象时,应该使用同步的集合类,例如,使用Vector代替ArrayList,用Hashtable代替HashMap