ServletContext 被 Servlet 程序用来与 Web 容器通信。例如写日志,转发请求。每一个 Web 应用程序含有一个Context,被Web应用内的各个程序共享。

因为Context可以用来保存资源并且共享,所以我所知道的 ServletContext 的最大应用是Web缓存----把不经常更改的内容读入内存,所以服务器响应请求的

时候就不需要进行慢速的磁盘I/O了。ServletContextListener 是 ServletContext 的监听者,如果 ServletContext 发生变化,如服务器启动时

ServletContext 被创建,服务器关闭时 ServletContext 将要被销毁。在JSP文件中,application 是 ServletContext 的实例,由JSP容器默认创建。

Servlet 中调用 getServletContext()方法得到 ServletContext 的实例。

我们使用缓存的思路大概是:

服务器启动时,ServletContextListener 的 contextInitialized()方法被调用,所以在里面创建好缓存。可以从文件中或者从数据库中读取取缓存内容

生成类,用 ervletContext.setAttribute()方法将缓存类保存在 ServletContext 的实例中。

程序使用 ServletContext.getAttribute()读取缓存。如果是 JSP,使用a pplication.getAttribute()。如果是 Servlet,使用

getServletContext().getAttribute()。如果缓存发生变化(如访问计数),你可以同时更改缓存和文件/数据库。或者你等 变化积累到一定程序再保存,也可以

在下一步保存。

服务器将要关闭时,ServletContextListener 的 contextDestroyed()方法被调用,所以在里面保存缓存的更改。将更改后的缓存保存回文件或者数据库,更

新原来的内容。

  import User; //my own class

  import DatabaseManager; // my own class

  import javax.servlet.ServletContext;

  import javax.servlet.ServletContextListener;

  public class MyContextListener

  implements ServletContextListener {

  private ServletContext context = null;

  public void contextInitialized(ServletContextEvent event) {

  context = event.getServletContext();

  User user = DatabaseManager.getUserById(1);

  context.setAttribute("user1", user);

  }

  public void contextDestroyed(ServletContextEvent event) {

  User user = (User)context.getAttribute("user1"); DatabaseManager.updateUserData(user);

  this.context = null;

  }

  }

  布署 ServletContextListener

  你实现(implements)了 ServletContextListener 编译后,把它放在正确的WEB-INF/classes目录下,更改WEB-INF目录下的 web.xml文件,在web-

app节点里添加<listener>

  <listener-class>MyServletContextListener</listener-class>

  </listener>

 下面的例子很好!!!

 在 Servlet API 中有一个 ServletContextListener 接口,它能够监听 ServletContext 对象的生命周期,实际上就是监听 Web 应用的生命周期。当

Servlet 容器启动或终止Web 应用时,会触发ServletContextEvent 事件,该事件由ServletContextListener 来处理。在 ServletContextListener 接口中

定义了处理ServletContextEvent 事件的两个方法。

  l contextInitialized(ServletContextEvent sce) :当Servlet 容器启动Web 应用时调用该方法。在调用完该方法之后,容器再对Filter 初始化,并且对

那些在Web 应用启动时就需要被初始化的Servlet 进行初始化。

  l contextDestroyed(ServletContextEvent sce) :当Servlet 容器终止Web 应用时调用该方法。在调用该方法之前,容器会先销毁所有的Servlet 和

Filter 过滤器。

  下面通过两个具体的例子来介绍 ServletContextListener 的用法。

  例一:在服务启动时,将数据库中的数据加载进内存,并将其赋值给一个属性名,其它的 Servlet 就可以通过 getAttribute 进行属性值的访问。有如下两

个步骤:1 : ServletContext 对象是一个为整个 web 应用提供共享的内存,任何请求都可以访问里面的内容

  2 :如何实现在服务启动的时候就动态的加入到里面的内容:我们需要做的有:

  1 ) 实现 servletContextListerner 接口 并将要通过 setAttribute ( name,data )方法提交到内存中去 ;

  2 )应用项目通过 getAttribute(name) 将数据取到 。

  package ServletContextTest;

  import java.sql.Connection;

  import java.sql.PreparedStatement;

  import java.sql.ResultSet;

  import java.util.HashMap;

  import java.util.Map;

  import javax.servlet.ServletContext;

  import javax.servlet.ServletContextEvent;

  import javax.servlet.ServletContextListener;

  import util.ConnectTool;

  public class ServletContextLTest implements ServletContextListener{

  // 实现其中的销毁函数

  public void contextDestroyed(ServletContextEvent sce) { System.out.println("this is last destroyeed");

  }

  // 实现其中的初始化函数,当有事件发生时即触发

  public void contextInitialized(ServletContextEvent sce) {

  ServletContext sct=sce.getServletContext();

  Map<Integer,String> depts=new HashMap<Integer,String>();

  Connection connection=null;

  PreparedStatement pstm=null;

  ResultSet rs=null;

  try{

  connection=ConnectTool.getConnection();

  String sql="select deptNo,dname from dept"; pstm=connection.prepareStatement(sql);

  rs=pstm.executeQuery();

  while(rs.next()){

  depts.put(rs.getInt(1), rs.getString(2));

  }

  // 将所取到的值存放到一个属性键值对中

  sct.setAttribute("dept", depts);

  System.out.println("======listener test is beginning=========");

  }catch(Exception e){

  e.printStackTrace();

  }finally{

  ConnectTool.releasersc(rs, pstm, connection);

  }

  }

  }

  在完成上述编码后,仍需在 web.xml 中进行如下配置,以使得该监听器可以起作用。<listener>

  <listener-class>ServletContextTest.ServletContextLTest</listener-class>

  </listener> 在完成上述配置后, web 服务器在启动时,会直接加载该监听器,通过以下的应用程序就可以进行数据的访问。

  package ServletContextTest; import java.io.IOException;

  import java.io.PrintWriter;

  import java.util.*;

  import javax.servlet.ServletContext;

  import javax.servlet.ServletException;

  import javax.servlet.http.HttpServlet;

  import javax.servlet.http.HttpServletRequest;

  import javax.servlet.http.HttpServletResponse;

  public class CreateEmployee extends HttpServlet{

  @Override

  protected void service(HttpServletRequest request, HttpServletResponse response)

  throws ServletException, IOException {

  ServletContext sct=getServletConfig().getServletContext();

  // 从上下文环境中通过属性名获取属性值

  Map<Integer,String> dept(Map<Integer,String>)sct.getAttribute("dept");

  Set<Integer> key=dept.keySet();

  response.setContentType("text/html;charset=utf-8");

  PrintWriter out=response.getWriter();

  out.println("<html>");

  out.println("<body>");

  out.println("<form action='/register' action='post'>");

  out.println("<table alignb='center'>");

  out.println("<tr>");

  out.println("<td>");

  out.println("username:");

  out.println("</td>");

  out.println("<td>");

  out.println("<input type='text' name='username'");

  out.println("</tr>");

  out.println("<tr>");

  out.println("<td>");

  out.println("city:");

  out.println("</td>");

  out.println("<td>");

  out.println("<select name='dept'");

  for(Integer i:key){

  out.println("<option value='"+i+"'>"+dept.get(i)+"</option>");

  }

  out.println("</select>");

  out.println("</td>");

  out.println("<tr>");

  out.println("</table>");

  out.println("</form>");

  out.println("</body>");

  out.println("</html>");

  out.flush();

  }

  }

  例二:书写一个类用于统计当Web 应用启动后,网页被客户端访问的次数。如果重新启动Web 应用,计数器不会重新从1 开始统计访问次数,而是从上次

统计的结果上进行累加。在实际应用中,往往需要统计自Web 应用被发布后网页被客户端访问的次数,这就要求当Web 应用被终止时,计数器的数值被永久存

储在一个文件中或者数据库中,等到Web 应用重新启动时,先从文件或数据库中读取计数器的初始值,然后在此基础上继续计数。

  向文件中写入或读取计数器的数值的功能可以由自定义的 MyServletContextListener 类来完成,它具有以下功能:

  1 、在 Web 应用启动时从文件中读取计数器的数值,并把表示计数器的 Counter 对象存放到 Web 应用范围内。存放计数器的文件的路径为

helloapp/count/count.txt 。

  2 、在Web 应用终止时把Web 应用范围内的计数器的数值保存到count.txt 文件中。package ServletContextTest; import

javax.servlet.ServletContext;

  import javax.servlet.ServletContextEvent;

  import javax.servlet.ServletContextListener; public class MyServletContextListener implements ServletContextListener{

  public void contextInitialized(ServletContextEvent sce){

  System.out.println("helloapp application is Initialized.");

  // 获取 ServletContext 对象

  ServletContext context=sce.getServletContext();

  try{

  // 从文件中读取计数器的数值

  BufferedReader reader=new BufferedReader(

  new InputStreamReader(context.

  getResourceAsStream("/count/count.txt")));

  int count=Integer.parseInt(reader.readLine());

  reader.close();

  // 创建计数器对象

  Counter counter=new Counter(count);

  // 把计数器对象保存到 Web 应用范围 context.setAttribute("counter",counter);

  } catch(IOException e) {

  e.printStackTrace();

  }

  }

  public void contextDestroyed(ServletContextEvent sce){

  System.out.println("helloapp application is Destroyed.");

  // 获取 ServletContext 对象

  ServletContext context=sce.getServletContext();

  // 从 Web 应用范围获得计数器对象

  Counter counter=(Counter)context.getAttribute("counter");

  if(counter!=null){

  try{

  // 把计数器的数值写到 count.txt 文件中

  String filepath=context.getRealPath("/count");

  filepath=filepath+"/count.txt";

  PrintWriter pw=new PrintWriter(filepath);

  pw.println(counter.getCount());

  pw.close();

  } catch(IOException e) {

  e.printStackTrace();

  }

  }

  }

  }

  将用户自定义的 MyServletContextListener 监听器在 Servlet 容器进行注册, Servlet 容器会在启动或终止 Web 应用时,会调用该监听器的相关方

法。在 web.xml 文件中, <listener> 元素用于向容器注册监听器:

  <listener>

  <listener-class> ServletContextTest .MyServletContextListener<listener-class /></listener>

  注意:<listener-class>中要写出类的包名,比如下面的写法(包名标识了类的路径)<listener>

  <listener-class>com.efarm360.action.StartupAction</listener-class>

  </listener>

  通过上述两个例子,即可以非常清楚的了解到 ServletContextListener 接口的使用方法及技巧。在Container 加载Web 应用程序时(例如启动

Container 之后),会呼叫contextInitialized() ,而当容器移除Web 应用程序时,会呼叫contextDestroyed () 方法。 通过 Tomcat 控制台的打印结果的

先后顺序,会发现当 Web 应用启动时,Servlet 容器先调用contextInitialized() 方法,再调用lifeInit 的init() 方法;当Web 应用终止时,Servlet 容器先调

用lifeInit 的destroy() 方法,再调用contextDestroyed() 方法。由此可见,在Web 应用的生命周期中,ServletContext 对象最早被创建,最晚被销毁。

技术分享:凯哥学堂

Ajax解决IE浏览器兼容问题的更多相关文章

  1. 使用一行代码解决IE浏览器兼容问题

    在网站开发中不免因为各种兼容问题苦恼,针对兼容问题,其实IE给出了解决方案Google也给出了解决方案 百度也应用了这种方案去解决IE的兼容问题 百度源代码如下 <!Doctype html&g ...

  2. 原生js解决跨浏览器兼容问题

    //跨浏览器兼容问题 Util = { //添加类名 add : function(ele,type,hand){ if(ele.addEventListener){ ele.addEventList ...

  3. 解决让浏览器兼容ES6特性

    为什么ES6会有兼容性问题? 由于广大用户使用的浏览器版本在发布的时候也许早于ES6的定稿和发布,而到了今天,我们在编程中如果使用了ES6的新特性,浏览器若没有更新版本,或者新版本中没有对ES6的特性 ...

  4. 解决IE浏览器兼容问题的一行代码

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...

  5. WEB前端开发人员须知的常见浏览器兼容问题及解决技巧

    所谓的浏览器兼容性问题,是指因为不同的浏览器对同一段代码有不同的解析,造成页面显示效果不统一的情况.在大多数情况下,我们的需求是,无论用户用什么浏览器来查看我们的网站或者登陆我们的系统,都应该是统一的 ...

  6. Ajax在IE浏览器会出现中文乱码解决办法

    在AJAX浏览器来进行发送数据时,一般它所默认的都是UTF-8的编码. Ajax在IE浏览器会出现中文乱码的情况!解决办法如下 <script type="text/javascrip ...

  7. IE内嵌google chrome frame解决浏览器兼容问题

    IE内嵌google chrome frame解决浏览器兼容问题  http://www.cnblogs.com/xwdreamer/archive/2013/12/17/3477776.html 参 ...

  8. 解决各大浏览器兼容问题hack

    解决各大浏览器兼容问题hack,IE6/ IE7/ IE8/ IE9/ Firefox/ Opera/ Webkit/ Chrome/ Safari. 浏览器兼容是网站前端页面制作最基本的问题,通常I ...

  9. 解决IE浏览器缓存导致AJAX请求数据异常

    IE10浏览器会把AJAX请求的数据都缓存下来,然后每次想去刷新数据时发现数据都是一样的,于是导致数据显示异常. 解决方法: 在页面<head>标签里,加上以下声明: <!-- 解决 ...

随机推荐

  1. poj 3080 Blue Jeans 解题报告

    题目链接:http://poj.org/problem?id=3080 该题属于字符串处理中的串模式匹配问题.题目要求我们:给出一个DNA碱基序列,输出最长的相同的碱基子序列.(保证在所有的序列中都有 ...

  2. Ubuntu 13.04安装搜狗输入法

    Ubuntu 13.04安装搜狗输入法 [日期:2013-07-08] 来源:Linux公社  作者:LinuxIDC.com [字体:大 中 小]     目标:在Ubuntu 13.04以及基于U ...

  3. Intellj IDEA快捷键

    Alt+回车 导入包,自动修正 Ctrl+N   查找类 Ctrl+Shift+N 查找文件 Ctrl+Alt+L  格式化代码 Ctrl+Alt+O 优化导入的类和包 Alt+Insert 生成代码 ...

  4. Diablo3

    1.装备 主手:元素弓 副手:精细箭袋 头: 胸:燃火外套 手:娜塔亚的手感 护腕:稳击护腕 戒指:罗盘玫瑰+布尔凯索的婚戒 颈部:旅者之誓 腰:科雷姆的强力腰带(速度加25%) 腿:深渊挖掘裤 脚: ...

  5. 创建型模式之Singleton模式

    单例模式大概是最直观的一种设计模式了,尽管直观却不简单. 数学与逻辑学中,singleton定义为“有且仅有一个元素的集合”, 单例模式可以如下定义:“一个类有且仅有一个实例,并且自行实例化向整个系统 ...

  6. ytu 1980:小鼠迷宫问题(DFS 深度优先搜索)

     小鼠迷宫问题 Time Limit: 2 Sec  Memory Limit: 64 MB Submit: 1  Solved: 1 [Submit][Status][Web Board] Desc ...

  7. matlab练习程序(射线法判断点与多边形关系)

    依然是计算几何. 射线法判断点与多边形关系原理如下: 从待判断点引出一条射线,射线与多边形相交,如果交点为偶数,则点不在多边形内,如果交点为奇数,则点在多边形内. 原理虽是这样,有些细节还是要注意一下 ...

  8. loj 1031(区间dp+记忆化搜索)

    题目链接:http://lightoj.com/volume_showproblem.php?problem=1031 思路:dp[i][j]表示从区间i-j中能取得的最大值,然后就是枚举分割点了. ...

  9. [转]关于int整形变量占有字节问题

    int的长度由处理器(16位,32位,64位)和比哪一期决定. 首先从处理器来讲 :16位处理器中的int 占有16位 即2个字节                         32位处理器中int ...

  10. codeforces 385 c

    Description Recently, the bear started studying data structures and faced the following problem. You ...