ServletContext可以初始化String类型的参数。但是,如果你希望应用初始化参数是一个数据库DataSource呢?
上下文参数只能是String。毕竟,你不能把一个Dog对象塞到XML部署描述文件中(事实上,可以用XML表示一个串行化对象,但是在当前的Servlet规范中还没有相关的支持……没准将来会提供)。
如果你真的想让Web应用的所有部分都能访问一个共享的数据连接,该怎么办?
当然可以把这个DataSource查找名放在一个上下文初始化参数里,这也是当前上下文参数最常见的一种用法。
不过,之后谁将这个String参数转换成由Web应用各部分共享的一个具体DataSource引用呢?
不能把这个代码放在servlet中,因为你选择谁作为第一个servlet来查找DataSource并把它存储在一个属性中呢?你真的想保证总是让某个特定的servlet最先运行吗?好好考虑一下。
解决办法:
可以建立一个单独的类,而不是一个servlet或JSP,它能监听ServletContext一生中的两个关键事件:初始化(创建)和撤销。这个类实现了javax.servlet.ServletContextListener。
ServletContextListener类:
package com.moonlit; import javax.servlet.ServletContextEvent; import javax.servlet.ServletContextListener; public class MyServletContextListener implements ServletContextListener { @Override public void contextDestroyed(ServletContextEvent event) { // 初始化数据库连接的代码 // and store it as a context attribute } @Override public void contextInitialized(ServletContextEvent event) { // 关闭数据库连接的代码 } }
教程:一个简单的ServletContextListener
在这个例子中,我们要把String初始化参数转换成一个真正的对象——一个Dog。监听者的任务是得到有关狗的品种的上下文初始化参数(Beagle、Poodle、等等),然后使用这个String来构造一个Dog对象。监听者再把这个Dog对象存到一个ServletContext属性中,以便servlet获取。
关键是,servlet现在能访问一个共享的数据对象(在这里,就是一个共享的Dog),而且不用读上下文对象。这个共享的对象是一个Dog还是一个数据库连接并没有关系。重点是要使用初始化参数来创建一个对象,让应用的所有部分都能共享这个对象。
这个例子将包括如下过程:
□ 监听者对象向ServletContextEvent对象请求应用ServletContext对象的一个引用。
□ 监听者使用这个ServletContext引用得到"breed"上下文初始化参数,这是一个String,表示狗的品种。
□ 监听者使用这个狗的品种String构造一个Dog对象。
□ 监听者使用ServletContext引用在ServletContext中设置Dog属性。
□ 这个Web应用的测试servlet从ServletContext得到Dog对象,并调用这个Dog的getBreed()方法。
建立和使用一个上下文监听者
要建立和使用一个上下文监听者,我们需要在web.xml中部署描述文件,在web.xml中添加如下内容:
<listener> <listener-class> com.moonlit.MyServletContextListener </listener-class> </listener>
除了设置ServletContextListener,我们还要设置ServletContext的初始化参数,以及测试的servlet的相关的参数。
完整的web.xml如下:
<?xml version="1.0" encoding="UTF-8"?> <web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://xmlns.jcp.org/xml/ns/javaee" xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd" id="WebApp_ID" version="3.1"> <display-name>DogListenerExample</display-name> <servlet> <servlet-name>ListenerTester</servlet-name> <servlet-class>com.moonlit.ListenerTester</servlet-class> </servlet> <servlet-mapping> <servlet-name>ListenerTester</servlet-name> <url-pattern>/ListenerTester.do</url-pattern> </servlet-mapping> <context-param> <param-name>breed</param-name> <param-value>Great Dane</param-value> </context-param> <listener> <listener-class> com.moonlit.MyServletContextListener </listener-class> </listener> </web-app>
MyServletContextListener是监听者类。这个类实现了ServletContextListener,它得到上下文初始化参数,创建Dog,并把Dog作为一个上下文属性。
MyServlerContextListener.java:
package com.moonlit; import javax.servlet.ServletContext; import javax.servlet.ServletContextEvent; import javax.servlet.ServletContextListener; public class MyServletContextListener implements ServletContextListener { @Override public void contextInitialized(ServletContextEvent event) { ServletContext sc = event.getServletContext(); String dogBreed = sc.getInitParameter("breed"); Dog d = new Dog(dogBreed); sc.setAttribute("dog", d); } @Override public void contextDestroyed(ServletContextEvent event) { // do nothing here } }
我们需要建立一个属性类Dog。DOg只是一个普通的Java类,它的任务是作为一个属性值,由ServletContextListener实例化,并设置在ServletContext中,一遍servlet获取。
Dog.java:
package com.moonlit; public class Dog { private String breed; public Dog(String breed) { this.breed = breed; } public String getBreed() { return breed; } }
ListenerTest类是一个Servlet类,它的任务是验证监听者的工作。为此这个Servlet会从上下文得到Dog属性,调用Dog的getBreed(),把结果打印到响应(使我们能在浏览器中看到)。
ListenerTester.java:
package com.moonlit; 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; @SuppressWarnings("serial") public class ListenerTester extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { resp.setContentType("text/html"); PrintWriter out = resp.getWriter(); out.println("test context attributes set by listener<br>"); out.println("<br>"); Dog dog = (Dog) getServletContext().getAttribute("dog"); out.println("Dog's breed is : " + dog == null ? "no " : dog.getBreed()); } }
效果如下: