Servlet和JSP的一个重大的区别即是Servlet可以通过web.xml文件的配置让Servlet在Web容器启动时就自动启动Servlet。可以利用Servlet的这个特性不变化的数据事先加载到Web应用服务器中以便缓存使用。
假设,我们系统的用户在系统部署前就已经创建好,以后不常发生变化,那么我们可以在Web应用程序启动时就将其下载缓存到Web应用服务器内存中,如果用户发生变化可以手工调用这个Servlet进行刷新。下面我们就通过Servlet向导创建这个UserCacheServlet,它在Web容器启动时自动下载并缓存系统所有用户Id和用户名:
1.启动创建Servlet向导,填写Servlet名字
通过File->New...->Web->双击Standard Servlet图标启动创建Servlet向导的第一步,如下图所示:
图 2填写Servlet名字
在Class name中填入Servlet的名字:UserCacheServlet,在Package中填入bookstore.Servlet作为包名。按Next到下一步。
2.选择Servlet所要实现的方法
我们在前面已经介绍了Servlet通过不同的doXxx()方法的响应HTTP请求方式,你可以在向导的第2步选择需要定义哪些doXxx()方法。默认情况下doGet()方法被勾选,即通过HTTP GET请求方式访问Servlet。通过带参的URL访问Servlet时,Servlet就用doGet()方法响应这个请求。由于我们只是假设用户数据不常变动,并不是说永远不变动,所以我们在Web容器初始化时,希望通过UserCacheServlet自动加载用户数据到缓存中,当数据库表T_USER的用户数据发生变动时,我们可以手工调用UserCacheServlet,让其刷新缓存中的用户数据。
Web容器启动时自动初始化UserCacheServlet,此时init()方法被调用,我们可以通过init()方法加载用户数据,当用户通过URL请求刷新用户数据时,UserCacheServlet通过doGet()方法响应这个HTTP GET请求。也就是说,我们需要实现doGet()方法,所以我们接受向导的默认设置,如下图所示:
图 3 选择需要覆盖的Servlet响应方法
按Next跳过第3步到向导的第4步。
3.指定访问Servlet的路径
图 4 指定Servlet访问路径
·Name:usercacheServlet,Servlet在web.xml配置文件中所取的名字
·URL pattern:/usercacheServlet,访问这个Servlet的匹配路径。指定这个访问路径后,假设Web应用程序部署在下,则通过访问Servlet。
直接按Finish创建Servlet。
打开web.xml文件,你可以找到关于UserCacheServlet声明和访问的部署描述信息:
·<Servlet>节点:描述Servlet的名字及类名。
·<Servlet-mapping>节点:描述Servlet访问匹配路径。
双击工程窗格资源树的webModule节点,JBuilder在内容窗格中打开用于编辑web.xml文件的Web模块DD编辑器(Web Module DD editor),此时结构窗格显示出web.xml文件的结构,如下图所示:
图 5 结构窗格的web.xml文件结构树
带 图标的节点表示已经有配置内容,而未带 图标的节点表示暂时还没有对应的配置内容。我们展开Servlets节点,定位到usercacheServlet并双击这个节点,DD编辑器调整界面对usercacheServlet这个Servlet进行配置,如下图所示:
图 6 DD编辑器
Servlet可以在Web容器启动时,自动初始化。假设有多个Servlet都需要自动初始化,则可以通过web.xml的<load-on-startup>设置启动的次序。我们在DD编辑器将Load on startup值设为2, 这样UserCacheServlet将在Web容器启动后,排在顺序2初始化。
一些系统所用的Servlet,由于是Web开始服务的基础必须在顺序1初化,所以我们开发的Servlet最好在顺序2或顺序3初始化。Servlet初始化时,init()方法被调用。在进行这样的设置后web.xml将包含以下粗体所示的配置信息。
代码清单 1 web.xml有关UserCacheServlet的描述信息
1. <web-app>
2. …
3. <Servlet>
4. <Servlet-name>usercacheServlet</Servlet-name>
5. <Servlet-class>bookstore.Servlet.UserCacheServlet</Servlet-class>
6. <load-on-startup>2</load-on-startup>
7. </Servlet>
8. <Servlet-mapping>
9. <Servlet-name>usercacheServlet</Servlet-name>
10. <url-pattern>/usercacheServlet</url-pattern>
11. </Servlet-mapping>
12. …
13. </web-app>
注意:
当删除UserCacheServlet后,Servlet在web.xml所对应的部署描述信息并不会一起删除,你必须手工删除。
在init()初始化方法中利用UserList.fillUser()方法从数据库中下载并缓存用户记录信息,在doGet()方法中也引用了UserList.fillUser(),用户通过URL访问UserCacheServlet时,doGet()方法被调用,刷新缓存用户数据,并显示"刷新成功"提示,其代码如下所示:
代码清单 2 UserCacheServlet.java
1. package bookstore.Servlet;
2.
3. import javax.Servlet.*;
4. import javax.Servlet.http.*;
5. import java.io.*;
6. import bookstore.UserList;
7.
8. public class UserCacheServlet
9. extends HttpServlet
10. {
11. private static final String CONTENT_TYPE = "text/html; charset=GBK";
12.
13. //Initialize global variables
14. public void init()
15. throws ServletException
16. {
17. UserList.fillUser();//Web容器启动后调用
18. }
19.
20. //Process the HTTP Get request
21. public void doGet(HttpServletRequest request, HttpServletResponse response)
22. throws ServletException, IOException
23. {
24. UserList.fillUser();//刷新用户数据
25. response.setContentType(CONTENT_TYPE);
26. PrintWriter out = response.getWriter();
27. out.println("<html>");
28. out.println("<head><title>UserCacheServlet</title></head>");
29. out.println("<body bgcolor="#ffffff">");
30. out.println("刷新成功!");
31. out.println("</body>");
32. out.println("</html>");
33. out.close();
34. }
当然,我们要对《JBuilder 2005实战JSP开发》专题中创建的UserList.java代码进行更改,定义fillUser()方法以供UserCacheServlet.java调用,此外,还需要调整原getUserListHTML()方法,如下所示:
代码清单 3 调整后的UserList.java代码
2. package bookstore;
3.
4. import java.sql.*;
5. import java.util.*;
6.
7. public class UserList
8. {
9. private static Map userMap;//用户ID和用户名的Map
10. //将用户数据缓存到Map中
11. public static void fillUser()
12. {
13. if (userMap == null)
14. {
15. userMap = new HashMap();
16. } else
17. {
18. userMap.clear();
19. }
20. Connection conn = null;
21. StringBuffer sBuf = new StringBuffer();
22. try
23. {
24. conn = DBConnection.getConnection();
25. PreparedStatement pStat = conn.prepareStatement(
26. "select USER_ID,USER_NAME from T_USER");
27. ResultSet rs = pStat.executeQuery();
28. while (rs.next())
29. {
30. userMap.put(rs.getString(1), rs.getString(2));
31. }
32. } catch (SQLException ex)
33. {
34. ex.printStackTrace();
35. } finally
36. {
37. try
38. {
39. if (conn != null)
40. {
41. conn.close();
42. conn = null;
43. }
44. } catch (SQLException ex1)
45. {
46. ex1.printStackTrace();
47. }
48. }
49. }
50.
51. //获取HTML下拉框的用户列表代码
52. public static String getUserListHTML()
53. {
54. StringBuffer sBuf = new StringBuffer();
55. Set set = userMap.keySet();
56. Iterator iter = set.iterator(