Servlet 教程

Servlet 为创建基于 web 的应用程序提供了基于组件、独立于平台的方法,可以不受 CGI 程序的性能限制。Servlet 有权限访问所有的 Java API,包括访问企业级数据库的 JDBC API。
本教程将讲解如何使用 Java Servlet 来开发基于 web 的应用程序。

谁适合阅读本教程?

本教程是专为 Java 程序员设计的。在阅读本教程之前,需要先了解 Java Servlet 框架和它的 API。学习完本教程后,您会发现自己已经达到使用 Java Servlet 的中等水平,后续您可以自行通过更深入的学习和实践完成进阶。

阅读本教程前,您需要了解的知识:

在您开始阅读本教程之前,最好对 Java 编程语言有一个很好的理解。如果您对 web 应用程序和互联网如何工作的有基本的认识,将有助于您理解本教程。

Servlet 相关教程

Servlet 是什么?

Java Servlet 是运行在 Web 服务器或应用服务器上的程序,它是作为来自 Web 浏览器或其他 HTTP 客户端的请求和 HTTP 服务器上的数据库或应用程序之间的中间层。

使用 Servlet,您可以收集来自网页表单的用户输入,呈现来自数据库或者其他源的记录,还可以动态创建网页。

Java Servlet 通常情况下与使用 CGI(Common Gateway Interface,公共网关接口)实现的程序可以达到异曲同工的效果。但是相比于 CGI,Servlet 有以下几点优势:

  • 性能明显更好。
  • Servlet 在 Web 服务器的地址空间内执行。这样它就没有必要再创建一个单独的进程来处理每个客户端请求。
  • Servlet 是独立于平台的,因为它们是用 Java 编写的。
  • 服务器上的 Java 安全管理器执行了一系列限制,以保护服务器计算机上的资源。因此,Servlet 是可信的。
  • Java 类库的全部功能对 Servlet 来说都是可用的。它可以通过 sockets 和 RMI 机制与 applets、数据库或其他软件进行交互。

Servlet 架构

下图显示了 Servlet 在 Web 应用程序中的位置。

Servlet 架构

Servlet 任务

Servlet 执行以下主要任务:

  • 读取客户端(浏览器)发送的显式的数据。这包括网页上的 HTML 表单,或者也可以是来自 applet 或自定义的 HTTP 客户端程序的表单。
  • 读取客户端(浏览器)发送的隐式的 HTTP 请求数据。这包括 cookies、媒体类型和浏览器能理解的压缩格式等等。
  • 处理数据并生成结果。这个过程可能需要访问数据库,执行 RMI 或 CORBA 调用,调用 Web 服务,或者直接计算得出对应的响应。
  • 发送显式的数据(即文档)到客户端(浏览器)。该文档的格式可以是多种多样的,包括文本文件(HTML 或 XML)、二进制文件(GIF 图像)、Excel 等。
  • 发送隐式的 HTTP 响应到客户端(浏览器)。这包括告诉浏览器或其他客户端被返回的文档类型(例如 HTML),设置 cookies 和缓存参数,以及其他类似的任务。

Servlet 包

Java Servlet 是运行在带有支持 Java Servlet 规范的解释器的 web 服务器上的 Java 类。

Servlet 可以使用 javax.servletjavax.servlet.http 包创建,它是 Java 企业版的标准组成部分,Java 企业版是支持大型开发项目的 Java 类库的扩展版本。

这些类实现 Java Servlet 和 JSP 规范。在写本教程的时候,二者相应的版本分别是 Java Servlet 2.5 和 JSP 2.1。

Java Servlet 就像任何其他的 Java 类一样已经被创建和编译。在您安装 Servlet 包并把它们添加到您的计算机上的 Classpath 类路径中之后,您就可以通过 JDK 的 Java 编译器或任何其他编译器来编译 Servlet。

下一步呢?

接下来,本教程会带你一步一步地设置您的 Servlet 环境,以便开始后续的 Servlet 使用。因此,请系紧您的安全带,随我们一起开始 Servlet 的学习之旅吧!相信您会很喜欢这个教程的。

Servlet 环境设置

开发环境是您可以开发、测试、运行 Servlet 的地方。

就像任何其他的 Java 程序,您需要通过使用 Java 编译器 javac 编译 Servlet,在编译 Servlet 应用程序后,将它部署在配置的环境中以便测试和运行。

这个开发环境设置包括以下步骤:

设置 Java 开发工具包(Java Development Kit)

这一步涉及到下载 Java 软件开发工具包(SDK,即 Software Development Kit),并适当地设置 PATH 环境变量。

您可以从 Oracle 的 Java 网站下载 SDK:Java SE Downloads

一旦您下载了 SDK,请按照给定的指令来安装和配置设置。最后,设置 PATH 和 JAVA_HOME 环境变量指向包含 java 和 javac 的目录,通常分别为 java_install_dir/bin 和 java_install_dir。

如果您运行的是 Windows,并把 SDK 安装在 C:jdk1.5.0_20 中,则需要在您的 C:autoexec.bat 文件中放入下列的行:

set PATH=C:jdk1.5.0_20in;%PATH%set JAVA_HOME=C:jdk1.5.0_20

或者,在 Windows NT/2000/XP 中,您也可以用鼠标右键单击"我的电脑",选择"属性",再选择"高级","环境变量"。然后,更新 PATH 的值,按下"确定"按钮。

在 Unix(Solaris、Linux 等)上,如果 SDK 安装在 /usr/local/jdk1.5.0_20 中,并且您使用的是 C shell,则需要在您的 .cshrc 文件中放入下列的行:

setenv PATH /usr/local/jdk1.5.0_20/bin:$PATHsetenv JAVA_HOME /usr/local/jdk1.5.0_20

另外,如果您使用集成开发环境(IDE,即 Integrated Development Environment),比如 Borland JBuilder、Eclipse、IntelliJ IDEA 或 Sun ONE Studio,编译并运行一个简单的程序,以确认该 IDE 知道您安装的 Java 路径。

设置 Web 服务器:Tomcat

在市场上有许多 Web 服务器支持 Servlet。有些 Web 服务器是免费下载的,Tomcat 就是其中的一个。

Apache Tomcat 是一款 Java Servlet 和 JavaServer Pages 技术的开源软件实现,可以作为测试 Servlet 的独立服务器,而且可以集成到 Apache Web 服务器。下面是在电脑上安装 Tomcat 的步骤:

  • http://tomcat.apache.org/ 上下载最新版本的 Tomcat。
  • 一旦您下载了 Tomcat,解压缩到一个方便的位置。例如,如果您使用的是 Windows,则解压缩到 C:apache-tomcat-5.5.29 中,如果您使用的是 Linux/Unix,则解压缩到 /usr/local/apache-tomcat-5.5.29 中,并创建 CATALINA_HOME 环境变量指向这些位置。

在 Windows 上,可以通过执行下面的命令来启动 Tomcat:

 %CATALINA_HOME%instartup.bat or C:apache-tomcat-5.5.29instartup.bat

在 Unix(Solaris、Linux 等) 上,可以通过执行下面的命令来启动 Tomcat:

$CATALINA_HOME/bin/startup.shor/usr/local/apache-tomcat-5.5.29/bin/startup.sh

Tomcat 启动后,可以通过在浏览器地址栏输入 http://localhost:8080/ 访问 Tomcat 中的默认应用程序。如果一切顺利,那么会显示以下结果:

Tomcat Home page

有关配置和运行 Tomcat 的进一步信息可以查阅应用程序安装的文档,或者可以访问 Tomcat 网站:http://tomcat.apache.org

在 Windows 上,可以通过执行下面的命令来停止 Tomcat:

C:apache-tomcat-5.5.29inshutdown

在 Unix(Solaris、Linux 等) 上,可以通过执行下面的命令来停止 Tomcat:

/usr/local/apache-tomcat-5.5.29/bin/shutdown.sh

设置 CLASSPATH

由于 Servlet 不是 Java 平台标准版的组成部分,所以您必须为编译器指定 Servlet 类的路径。

如果您运行的是 Windows,则需要在您的 C:autoexec.bat 文件中放入下列的行:

set CATALINA=C:apache-tomcat-5.5.29set CLASSPATH=%CATALINA%commonlibservlet-api.jar;%CLASSPATH%

或者,在 Windows NT/2000/XP 中,您也可以用鼠标右键单击"我的电脑",选择"属性",再选择"高级","环境变量"。然后,更新 CLASSPATH 的值,按下"确定"按钮。

在 Unix(Solaris、Linux 等)上,如果您使用的是 C shell,则需要在您的 .cshrc 文件中放入下列的行:

setenv CATALINA=/usr/local/apache-tomcat-5.5.29setenv CLASSPATH $CATALINA/common/lib/servlet-api.jar:$CLASSPATH

注意:假设您的开发目录是 C:ServletDevel(在 Windows 上)或 /user/ServletDevel(在 UNIX 上),那么您还需要在 CLASSPATH 中添加这些目录,添加方式与上面的添加方式类似。

Servlet 生命周期

Servlet 生命周期可被定义为从创建直到毁灭的整个过程。以下是 Servlet 遵循的过程:

  • Servlet 通过调用 init () 方法进行初始化。
  • Servlet 调用 service() 方法来处理客户端的请求。
  • Servlet 通过调用 destroy() 方法终止(结束)。
  • 最后,Servlet 是由 JVM 的垃圾回收器进行垃圾回收的。

现在让我们详细讨论生命周期的方法。

init() 方法

init 方法被设计成只调用一次。它在第一次创建 Servlet 时被调用,在后续每次用户请求时不再调用。因此,它是用于一次性初始化,就像 Applet 的 init 方法一样。

Servlet 创建于用户第一次调用对应于该 Servlet 的 URL 时,但是您也可以指定 Servlet 在服务器第一次启动时被加载。

当用户调用一个 Servlet 时,就会创建一个 Servlet 实例,每一个用户请求都会产生一个新的线程,适当的时候移交给 doGet 或 doPost 方法。init() 方法简单地创建或加载一些数据,这些数据将被用于 Servlet 的整个生命周期。

init 方法的定义如下:

public void init() throws ServletException {  // 初始化代码...}

service() 方法

service() 方法是执行实际任务的主要方法。Servlet 容器(即 Web 服务器)调用 service() 方法来处理来自客户端(浏览器)的请求,并把格式化的响应写回给客户端。

每次服务器接收到一个 Servlet 请求时,服务器会产生一个新的线程并调用服务。service() 方法检查 HTTP 请求类型(GET、POST、PUT、DELETE 等),并在适当的时候调用 doGet、doPost、doPut,doDelete 等方法。

下面是该方法的特征:

public void service(ServletRequest request,                     ServletResponse response)       throws ServletException, IOException{}

service() 方法由容器调用,service 方法在适当的时候调用 doGet、doPost、doPut、doDelete 等方法。所以,您不用对 service() 方法做任何动作,您只需要根据来自客户端的请求类型来重载 doGet() 或 doPost() 即可。

doGet() 和 doPost() 方法是每次服务请求中最常用的方法。下面是这两种方法的特征。

doGet() 方法

GET 请求来自于一个 URL 的正常请求,或者来自于一个未指定 METHOD 的 HTML 表单,它由 doGet() 方法处理。

public void doGet(HttpServletRequest request,                  HttpServletResponse response)    throws ServletException, IOException {    // Servlet 代码}

doPost() 方法

POST 请求来自于一个特别指定了 METHOD 为 POST 的 HTML 表单,它由 doPost() 方法处理。

public void doPost(HttpServletRequest request,                   HttpServletResponse response)    throws ServletException, IOException {    // Servlet 代码}

destroy() 方法

destroy() 方法只会被调用一次,在 Servlet 生命周期结束时被调用。destroy() 方法可以让您的 Servlet 关闭数据库连接、停止后台线程、把 Cookie 列表或点击计数器写入到磁盘,并执行其他类似的清理活动。

在调用 destroy() 方法之后,servlet 对象被标记为垃圾回收。destroy 方法定义如下所示:

  public void destroy() {    // 终止化代码...  }

架构图

下图显示了一个典型的 Servlet 生命周期方案。

  • 第一个到达服务器的 HTTP 请求被委派到 Servlet 容器。
  • Servlet 容器在调用 service() 方法之前加载 Servlet。
  • 然后 Servlet 容器处理由多个线程产生的多个请求,每个线程执行一个单一的 Servlet 实例的 service() 方法。

Servlet 生命周期 

以下为生命周期的一个简单实例:

//servlet生命周期,的三个方法,1.被创建,执行且只执行一次init方法,2.提供服务,执行service方法,执行多次 3.被销毁,当Servlet服务器正常关闭时,执行destroy方法,只执行一次。          @Override     public void init() throws ServletException {         // TODO Auto-generated method stub         super.init();     }          @Override     protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {         // TODO Auto-generated method stub         super.service(req, resp);     }          @Override     public void destroy() {         // TODO Auto-generated method stub         super.destroy();     }


Servlet 实例

Servlet 是服务 HTTP 请求并实现 javax.servlet.Servlet 接口的 Java 类。Web 应用程序开发人员通常编写 Servlet 来扩展 javax.servlet.http.HttpServlet,并实现 Servlet 接口的抽象类专门用来处理 HTTP 请求。

Hello World 示例代码

下面是 Servlet 输出 Hello World 的示例源代码:

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*;// 扩展 HttpServlet 类public class HelloWorld extends HttpServlet {   private String message;  public void init() throws ServletException  {      // 执行必需的初始化      message = "Hello World";  }  public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 设置响应内容类型      response.setContentType("text/html");      // 实际的逻辑是在这里      PrintWriter out = response.getWriter();      out.println("<h1>" + message + "</h1>");  }    public void destroy()  {      // 什么也不做  }}

编译 Servlet

让我们把上面的代码写在 HelloWorld.java 文件中,把这个文件放在 C:ServletDevel(在 Windows 上)或 /usr/ServletDevel(在 UNIX 上)中,您还需要把这些目录添加到 CLASSPATH 中。

假设您的环境已经正确地设置,进入 ServletDevel 目录,并编译 HelloWorld.java,如下所示:

$ javac HelloWorld.java

如果 Servlet 依赖于任何其他库,您必须在 CLASSPATH 中包含那些 JAR 文件。在这里,我只包含了 servlet-api.jar JAR 文件,因为我没有在 Hello World 程序中使用任何其他库。

该命令行使用 Sun Microsystems Java 软件开发工具包(JDK)内置的 javac 编译器。为使该命令正常工作,您必须 PATH 环境变量中使用的 Java SDK 的位置。

如果一切顺利,上面编译会在同一目录下生成 HelloWorld.class 文件。下一节将讲解已编译的 Servlet 如何部署在生产中。

Servlet 部署

默认情况下,Servlet 应用程序位于路径 <Tomcat-installation-directory>/webapps/ROOT 下,且类文件放在 <Tomcat-installation-directory>/webapps/ROOT/WEB-INF/classes 中。

如果您有一个完全合格的类名称 com.myorg.MyServlet,那么这个 Servlet 类必须位于 WEB-INF/classes/com/myorg/MyServlet.class 中。

现在,让我们把 HelloWorld.class 复制到 <Tomcat-installation-directory>/webapps/ROOT/WEB-INF/classes 中,并在位于 <Tomcat-installation-directory>/webapps/ROOT/WEB-INF/ 的 web.xml 文件中创建以下条目:

    <servlet>        <servlet-name>HelloWorld</servlet-name>        <servlet-class>HelloWorld</servlet-class>    </servlet>    <servlet-mapping>        <servlet-name>HelloWorld</servlet-name>        <url-pattern>/HelloWorld</url-pattern>    </servlet-mapping>

上面的条目要被创建在 web.xml 文件中的 <web-app>...</web-app> 标签内。在该文件中可能已经有各种可用的条目,但不要在意。

到这里,您基本上已经完成了,现在让我们使用 <Tomcat-installation-directory>instartup.bat(在 Windows 上)或 <Tomcat-installation-directory>/bin/startup.sh(在 Linux/Solaris 等上)启动 tomcat 服务器,最后在浏览器的地址栏中输入 http://localhost:8080/HelloWorld。如果一切顺利,您会看到下面的结果:

Servlet 实例

Servlet 表单数据

很多情况下,需要传递一些信息,从浏览器到 Web 服务器,最终到后台程序。浏览器使用两种方法可将这些信息传递到 Web 服务器,分别为 GET 方法和 POST 方法。

GET 方法

GET 方法向页面请求发送已编码的用户信息。页面和已编码的信息中间用 ? 字符分隔,如下所示:

http://www.test.com/hello?key1=value1&key2=value2

GET 方法是默认的从浏览器向 Web 服务器传递信息的方法,它会产生一个很长的字符串,出现在浏览器的地址栏中。如果您要向服务器传递的是密码或其他的敏感信息,请不要使用 GET 方法。GET 方法有大小限制:请求字符串中最多只能有 1024 个字符。

这些信息使用 QUERY_STRING 头传递,并可以通过 QUERY_STRING 环境变量访问,Servlet 使用 doGet() 方法处理这种类型的请求。

POST 方法

另一个向后台程序传递信息的比较可靠的方法是 POST 方法。POST 方法打包信息的方式与 GET 方法基本相同,但是 POST 方法不是把信息作为 URL 中 ? 字符后的文本字符串进行发送,而是把这些信息作为一个单独的消息。消息以标准输出的形式传到后台程序,您可以解析和使用这些标准输出。Servlet 使用 doPost() 方法处理这种类型的请求。

使用 Servlet 读取表单数据

Servlet 处理表单数据,这些数据会根据不同的情况使用不同的方法自动解析:

  • getParameter():您可以调用 request.getParameter() 方法来获取表单参数的值。
  • getParameterValues():如果参数出现一次以上,则调用该方法,并返回多个值,例如复选框。
  • getParameterNames():如果您想要得到当前请求中的所有参数的完整列表,则调用该方法。

使用 URL 的 GET 方法实例

下面是一个简单的 URL,将使用 GET 方法向 HelloForm 程序传递两个值。

http://localhost:8080/HelloForm?first_name=ZARA&last_name=ALI

下面是处理 Web 浏览器输入的 HelloForm.java Servlet 程序。我们将使用 getParameter() 方法,可以很容易地访问传递的信息:

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*;// 扩展 HttpServlet 类public class HelloForm extends HttpServlet {   public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 设置响应内容类型      response.setContentType("text/html");      PrintWriter out = response.getWriter();      String title = "使用 GET 方法读取表单数据";      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +                "<html>
" +                "<head><title>" + title + "</title></head>
" +                "<body bgcolor="#f0f0f0">
" +                "<h1 align="center">" + title + "</h1>
" +                "<ul>
" +                "  <li><b>名字</b>:"                + request.getParameter("first_name") + "
" +                "  <li><b>姓氏</b>:"                + request.getParameter("last_name") + "
" +                "</ul>
" +                "</body></html>");  }}

假设您的环境已经正确地设置,编译 HelloForm.java,如下所示:

$ javac HelloForm.java

如果一切顺利,上述编译会产生 HelloForm.class 文件。接下来,您就必须把该类文件复制到 <Tomcat-installation-directory>/webapps/ROOT/WEB-INF/classes 中,并在位于 <Tomcat-installation-directory>/webapps/ROOT/WEB-INF/ 的 web.xml 文件中创建以下条目:

    <servlet>        <servlet-name>HelloForm</servlet-name>        <servlet-class>HelloForm</servlet-class>    </servlet>    <servlet-mapping>        <servlet-name>HelloForm</servlet-name>        <url-pattern>/HelloForm</url-pattern>    </servlet-mapping>

现在在浏览器的地址栏中输入 http://localhost:8080/HelloForm?first_name=ZARA&last_name=ALI ,并在触发上述命令之前确保已经启动 Tomcat 服务器。如果一切顺利,您会得到下面的结果:

使用 GET 方法读取表单数据

  • 名字:ZARA
  • 姓氏:ALI

使用表单的 GET 方法实例

下面是一个简单的实例,使用 HTML 表单和提交按钮传递两个值。我们将使用相同的 Servlet HelloForm 来处理输入。

<html><body><form action="HelloForm" method="GET">名字:<input type="text" name="first_name"><br />姓氏:<input type="text" name="last_name" /><input type="submit" value="提交" /></form></body></html>

保存这个 HTML 到 hello.htm 文件中,并把它放在 <Tomcat-installation-directory>/webapps/ROOT 目录下。当您访问 http://localhost:8080/Hello.htm 时,下面是上面表单的实际输出。


尝试输入名字和姓氏,然后点击"提交"按钮,在您本机上查看输出结果。基于所提供的输入,它会产生与上一个实例类似的结果。

使用表单的 POST 方法实例

让我们对上面的 Servlet 做小小的修改,以便它可以处理 GET 和 POST 方法。下面的 HelloForm.java Servlet 程序使用 GET 和 POST 方法处理由 Web 浏览器给出的输入。

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*;// 扩展 HttpServlet 类public class HelloForm extends HttpServlet {   // 处理 GET 方法请求的方法  public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 设置响应内容类型      response.setContentType("text/html");      PrintWriter out = response.getWriter();      String title = "Using GET Method to Read Form Data";      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +                "<html>
" +                "<head><title>" + title + "</title></head>
" +                "<body bgcolor="#f0f0f0">
" +                "<h1 align="center">" + title + "</h1>
" +                "<ul>
" +                "  <li><b>名字</b>:"                + request.getParameter("first_name") + "
" +                "  <li><b>姓氏</b>:"                + request.getParameter("last_name") + "
" +                "</ul>
" +                "</body></html>");  }  // 处理 POST 方法请求的方法  public void doPost(HttpServletRequest request,                     HttpServletResponse response)      throws ServletException, IOException {     doGet(request, response);  }}

现在,编译部署上述的 Servlet,并使用带有 POST 方法的 Hello.htm 进行测试,如下所示:

<html><body><form action="HelloForm" method="POST">名字:<input type="text" name="first_name"><br />姓氏:<input type="text" name="last_name" /><input type="submit" value="提交" /></form></body></html>

下面是上面表单的实际输出,尝试输入名字和姓氏,然后点击"提交"按钮,在您本机上查看输出结果。


基于所提供的输入,它会产生与上一个实例类似的结果。

将复选框数据传递到 Servlet 程序

当需要选择一个以上的选项时,则使用复选框。

下面是一个 HTML 代码实例 CheckBox.htm,一个带有两个复选框的表单。

<html><body><form action="CheckBox" method="POST" target="_blank"><input type="checkbox" name="maths" checked="checked" /> 数学<input type="checkbox" name="physics"  /> 物理<input type="checkbox" name="chemistry" checked="checked" />                                                 化学<input type="submit" value="选择学科" /></form></body></html>

这段代码的结果是下面的表单:


下面是 CheckBox.java Servlet 程序,处理 Web 浏览器给出的复选框输入。

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*;// 扩展 HttpServlet 类public class CheckBox extends HttpServlet {   // 处理 GET 方法请求的方法  public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 设置响应内容类型      response.setContentType("text/html");      PrintWriter out = response.getWriter();     String title = "读取复选框数据";      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +                "<html>
" +                "<head><title>" + title + "</title></head>
" +                "<body bgcolor="#f0f0f0">
" +                "<h1 align="center">" + title + "</h1>
" +                "<ul>
" +                "  <li><b>数学标识:</b>: "                + request.getParameter("maths") + "
" +                "  <li><b>物理标识:</b>: "                + request.getParameter("physics") + "
" +                "  <li><b>化学标识:</b>: "                + request.getParameter("chemistry") + "
" +                "</ul>
" +                "</body></html>");  }  // 处理 POST 方法请求的方法  public void doPost(HttpServletRequest request,                     HttpServletResponse response)      throws ServletException, IOException {     doGet(request, response);  }}

上面的实例将显示下面的结果:

读取复选框数据

  • 数学标识:on
  • 物理标识:null
  • 化学标识:on

读取所有的表单参数

以下是通用的实例,使用 HttpServletRequest 的 getParameterNames() 方法读取所有可用的表单参数。该方法返回一个枚举,其中包含未指定顺序的参数名。

一旦我们有一个枚举,我们可以以标准方式循环枚举,使用 hasMoreElements() 方法来确定何时停止,使用 nextElement() 方法来获取每个参数的名称。

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*;import java.util.*;// 扩展 HttpServlet 类public class ReadParams extends HttpServlet {   // 处理 GET 方法请求的方法  public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 设置响应内容类型      response.setContentType("text/html");      PrintWriter out = response.getWriter();     String title = "读取所有的表单数据";      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +        "<html>
" +        "<head><title>" + title + "</title></head>
" +        "<body bgcolor="#f0f0f0">
" +        "<h1 align="center">" + title + "</h1>
" +        "<table width="100%" border="1" align="center">
" +        "<tr bgcolor="#949494">
" +        "<th>参数名称</th><th>参数值</th>
"+        "</tr>
");      Enumeration paramNames = request.getParameterNames();            while(paramNames.hasMoreElements()) {         String paramName = (String)paramNames.nextElement();         out.print("<tr><td>" + paramName + "</td>
<td>");         String[] paramValues =                request.getParameterValues(paramName);         // 读取单个值的数据         if (paramValues.length == 1) {           String paramValue = paramValues[0];           if (paramValue.length() == 0)             out.println("<i>No Value</i>");           else             out.println(paramValue);         } else {             // 读取多个值的数据             out.println("<ul>");             for(int i=0; i < paramValues.length; i++) {                 out.println("<li>" + paramValues[i]);             }             out.println("</ul>");         }      }      out.println("</tr>
</table>
</body></html>");  }  // 处理 POST 方法请求的方法  public void doPost(HttpServletRequest request,                     HttpServletResponse response)      throws ServletException, IOException {     doGet(request, response);  }}

现在,通过下面的表单尝试上面的 Servlet:

<html><body><form action="ReadParams" method="POST" target="_blank"><input type="checkbox" name="maths" checked="checked" /> 数学<input type="checkbox" name="physics"  /> 物理<input type="checkbox" name="chemistry" checked="checked" /> 化学<input type="submit" value="选择学科" /></form></body></html>

现在使用上面的表单调用 Servlet,将产生以下结果:

读取所有的表单数据

参数名称参数值
maths on
chemistry on

您可以尝试使用上面的 Servlet 来读取其他的表单数据,比如文本框、单选按钮或下拉框等。

Servlet 客户端 HTTP 请求

当浏览器请求网页时,它会向 Web 服务器发送特定信息,这些信息不能被直接读取,因为这些信息是作为 HTTP 请求的头的一部分进行传输的。您可以查看 HTTP 协议 了解更多相关信息。

以下是来自于浏览器端的重要头信息,您可以在 Web 编程中频繁使用:

头信息描述
Accept这个头信息指定浏览器或其他客户端可以处理的 MIME 类型。值 image/pngimage/jpeg 是最常见的两种可能值。
Accept-Charset这个头信息指定浏览器可以用来显示信息的字符集。例如 ISO-8859-1。
Accept-Encoding这个头信息指定浏览器知道如何处理的编码类型。值 gzipcompress 是最常见的两种可能值。
Accept-Language这个头信息指定客户端的首选语言,在这种情况下,Servlet 会产生多种语言的结果。例如,en、en-us、ru 等。
Authorization这个头信息用于客户端在访问受密码保护的网页时识别自己的身份。
Connection这个头信息指示客户端是否可以处理持久 HTTP 连接。持久连接允许客户端或其他浏览器通过单个请求来检索多个文件。值 Keep-Alive 意味着使用了持续连接。
Content-Length这个头信息只适用于 POST 请求,并给出 POST 数据的大小(以字节为单位)。
Cookie这个头信息把之前发送到浏览器的 cookies 返回到服务器。
Host这个头信息指定原始的 URL 中的主机和端口。
If-Modified-Since这个头信息表示只有当页面在指定的日期后已更改时,客户端想要的页面。如果没有新的结果可以使用,服务器会发送一个 304 代码,表示 Not Modified 头信息。
If-Unmodified-Since这个头信息是 If-Modified-Since 的对立面,它指定只有当文档早于指定日期时,操作才会成功。
Referer这个头信息指示所指向的 Web 页的 URL。例如,如果您在网页 1,点击一个链接到网页 2,当浏览器请求网页 2 时,网页 1 的 URL 就会包含在 Referer 头信息中。
User-Agent这个头信息识别发出请求的浏览器或其他客户端,并可以向不同类型的浏览器返回不同的内容。

读取 HTTP 头的方法

下面的方法可用在 Servlet 程序中读取 HTTP 头。这些方法通过 HttpServletRequest 对象可用。

序号方法 & 描述
1Cookie[] getCookies()
返回一个数组,包含客户端发送该请求的所有的 Cookie 对象。
2Enumeration getAttributeNames()
返回一个枚举,包含提供给该请求可用的属性名称。
3Enumeration getHeaderNames()
返回一个枚举,包含在该请求中包含的所有的头名。
4Enumeration getParameterNames()
返回一个 String 对象的枚举,包含在该请求中包含的参数的名称。
5HttpSession getSession()
返回与该请求关联的当前 session 会话,或者如果请求没有 session 会话,则创建一个。
6HttpSession getSession(boolean create)
返回与该请求关联的当前 HttpSession,或者如果没有当前会话,且创建是真的,则返回一个新的 session 会话。
7Locale getLocale()
基于 Accept-Language 头,返回客户端接受内容的首选的区域设置。
8Object getAttribute(String name)
以对象形式返回已命名属性的值,如果没有给定名称的属性存在,则返回 null。
9ServletInputStream getInputStream()
使用 ServletInputStream,以二进制数据形式检索请求的主体。
10String getAuthType()
返回用于保护 Servlet 的身份验证方案的名称,例如,"BASIC" 或 "SSL",如果JSP没有受到保护则返回 null。
11String getCharacterEncoding()
返回请求主体中使用的字符编码的名称。
12String getContentType()
返回请求主体的 MIME 类型,如果不知道类型则返回 null。
13String getContextPath()
返回指示请求上下文的请求 URI 部分。
14String getHeader(String name)
以字符串形式返回指定的请求头的值。
15String getMethod()
返回请求的 HTTP 方法的名称,例如,GET、POST 或 PUT。
16String getParameter(String name)
以字符串形式返回请求参数的值,或者如果参数不存在则返回 null。
17String getPathInfo()
当请求发出时,返回与客户端发送的 URL 相关的任何额外的路径信息。
18String getProtocol()
返回请求协议的名称和版本。
19String getQueryString()
返回包含在路径后的请求 URL 中的查询字符串。
20String getRemoteAddr()
返回发送请求的客户端的互联网协议(IP)地址。
21String getRemoteHost()
返回发送请求的客户端的完全限定名称。
22String getRemoteUser()
如果用户已通过身份验证,则返回发出请求的登录用户,或者如果用户未通过身份验证,则返回 null。
23String getRequestURI()
从协议名称直到 HTTP 请求的第一行的查询字符串中,返回该请求的 URL 的一部分。
24String getRequestedSessionId()
返回由客户端指定的 session 会话 ID。
25String getServletPath()
返回调用 JSP 的请求的 URL 的一部分。
26String[] getParameterValues(String name)
返回一个字符串对象的数组,包含所有给定的请求参数的值,如果参数不存在则返回 null。
27boolean isSecure()
返回一个布尔值,指示请求是否使用安全通道,如 HTTPS。
28int getContentLength()
以字节为单位返回请求主体的长度,并提供输入流,或者如果长度未知则返回 -1。
29int getIntHeader(String name)
返回指定的请求头的值为一个 int 值。
30int getServerPort()
返回接收到这个请求的端口号。

HTTP Header 请求实例

下面的实例使用 HttpServletRequest 的 getHeaderNames() 方法读取 HTTP 头信息。该方法返回一个枚举,包含与当前的 HTTP 请求相关的头信息。

一旦我们有一个枚举,我们可以以标准方式循环枚举,使用 hasMoreElements() 方法来确定何时停止,使用 nextElement() 方法来获取每个参数的名称。

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*;import java.util.*; // 扩展 HttpServlet 类public class DisplayHeader extends HttpServlet {   // 处理 GET 方法请求的方法  public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 设置响应内容类型      response.setContentType("text/html");       PrintWriter out = response.getWriter();    String title = "HTTP Header 请求实例";      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +        "<html>
" +        "<head><title>" + title + "</title></head>
"+        "<body bgcolor="#f0f0f0">
" +        "<h1 align="center">" + title + "</h1>
" +        "<table width="100%" border="1" align="center">
" +        "<tr bgcolor="#949494">
" +        "<th>Header 名称</th><th>Header 值</th>
"+        "</tr>
");       Enumeration headerNames = request.getHeaderNames();            while(headerNames.hasMoreElements()) {         String paramName = (String)headerNames.nextElement();         out.print("<tr><td>" + paramName + "</td>
");         String paramValue = request.getHeader(paramName);         out.println("<td> " + paramValue + "</td></tr>
");      }      out.println("</table>
</body></html>");  }  // 处理 POST 方法请求的方法  public void doPost(HttpServletRequest request,                     HttpServletResponse response)      throws ServletException, IOException {     doGet(request, response);  }}

现在,调用上面的 Servlet 会产生以下结果:

HTTP Header 请求实例

Header 名称Header 值
accept */*
accept-language en-us
user-agent Mozilla/4.0 (compatible; MSIE 7.0; Windows NT 5.1; Trident/4.0; InfoPath.2; MS-RTC LM 8)
accept-encoding gzip, deflate
host localhost:8080
connection Keep-Alive
cache-control no-cache

Servlet 服务器 HTTP 响应

正如前面的章节中讨论的那样,当一个 Web 服务器响应一个 HTTP 请求时,响应通常包括一个状态行、一些响应报头、一个空行和文档。一个典型的响应如下所示:

HTTP/1.1 200 OKContent-Type: text/htmlHeader2: ......HeaderN: ...  (Blank Line)<!doctype ...><html><head>...</head><body>...</body></html>

状态行包括 HTTP 版本(在本例中为 HTTP/1.1)、一个状态码(在本例中为 200)和一个对应于状态码的短消息(在本例中为 OK)。

下表总结了从 Web 服务器端返回到浏览器的最有用的 HTTP 1.1 响应报头,您会在 Web 编程中频繁地使用它们:

头信息描述
Allow这个头信息指定服务器支持的请求方法(GET、POST 等)。
Cache-Control这个头信息指定响应文档在何种情况下可以安全地缓存。可能的值有:public、privateno-cache 等。Public 意味着文档是可缓存,Private 意味着文档是单个用户私用文档,且只能存储在私有(非共享)缓存中,no-cache 意味着文档不应被缓存。
Connection这个头信息指示浏览器是否使用持久 HTTP 连接。值 close 指示浏览器不使用持久 HTTP 连接,值 keep-alive 意味着使用持久连接。
Content-Disposition这个头信息可以让您请求浏览器要求用户以给定名称的文件把响应保存到磁盘。
Content-Encoding在传输过程中,这个头信息指定页面的编码方式。
Content-Language这个头信息表示文档编写所使用的语言。例如,en、en-us、ru 等。
Content-Length这个头信息指示响应中的字节数。只有当浏览器使用持久(keep-alive)HTTP 连接时才需要这些信息。
Content-Type这个头信息提供了响应文档的 MIME(Multipurpose Internet Mail Extension)类型。
Expires这个头信息指定内容过期的时间,在这之后内容不再被缓存。
Last-Modified这个头信息指示文档的最后修改时间。然后,客户端可以缓存文件,并在以后的请求中通过 If-Modified-Since 请求头信息提供一个日期。
Location这个头信息应被包含在所有的带有状态码的响应中。在 300s 内,这会通知浏览器文档的地址。浏览器会自动重新连接到这个位置,并获取新的文档。
Refresh这个头信息指定浏览器应该如何尽快请求更新的页面。您可以指定页面刷新的秒数。
Retry-After这个头信息可以与 503(Service Unavailable 服务不可用)响应配合使用,这会告诉客户端多久就可以重复它的请求。
Set-Cookie这个头信息指定一个与页面关联的 cookie。

设置 HTTP 响应报头的方法

下面的方法可用于在 Servlet 程序中设置 HTTP 响应报头。这些方法通过 HttpServletResponse 对象可用。

序号方法 & 描述
1String encodeRedirectURL(String url)
为 sendRedirect 方法中使用的指定的 URL 进行编码,或者如果编码不是必需的,则返回 URL 未改变。
2String encodeURL(String url)
对包含 session 会话 ID 的指定 URL 进行编码,或者如果编码不是必需的,则返回 URL 未改变。
3boolean containsHeader(String name)
返回一个布尔值,指示是否已经设置已命名的响应报头。
4boolean isCommitted()
返回一个布尔值,指示响应是否已经提交。
5void addCookie(Cookie cookie)
把指定的 cookie 添加到响应。
6void addDateHeader(String name, long date)
添加一个带有给定的名称和日期值的响应报头。
7void addHeader(String name, String value)
添加一个带有给定的名称和值的响应报头。
8void addIntHeader(String name, int value)
添加一个带有给定的名称和整数值的响应报头。
9void flushBuffer()
强制任何在缓冲区中的内容被写入到客户端。
10void reset()
清除缓冲区中存在的任何数据,包括状态码和头。
11void resetBuffer()
清除响应中基础缓冲区的内容,不清除状态码和头。
12void sendError(int sc)
使用指定的状态码发送错误响应到客户端,并清除缓冲区。
13void sendError(int sc, String msg)
使用指定的状态发送错误响应到客户端。
14void sendRedirect(String location)
使用指定的重定向位置 URL 发送临时重定向响应到客户端。
15void setBufferSize(int size)
为响应主体设置首选的缓冲区大小。
16void setCharacterEncoding(String charset)
设置被发送到客户端的响应的字符编码(MIME 字符集)例如,UTF-8。
17void setContentLength(int len)
设置在 HTTP Servlet 响应中的内容主体的长度,该方法设置 HTTP Content-Length 头。
18void setContentType(String type)
如果响应还未被提交,设置被发送到客户端的响应的内容类型。
19void setDateHeader(String name, long date)
设置一个带有给定的名称和日期值的响应报头。
20void setHeader(String name, String value)
设置一个带有给定的名称和值的响应报头。
21void setIntHeader(String name, int value)
设置一个带有给定的名称和整数值的响应报头。
22void setLocale(Locale loc)
如果响应还未被提交,设置响应的区域。
23void setStatus(int sc)
为该响应设置状态码。

HTTP Header 响应实例

您已经在前面的实例中看到 setContentType() 方法,下面的实例也使用了同样的方法,此外,我们会用 setIntHeader() 方法来设置 Refresh 头。

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*;import java.util.*; // 扩展 HttpServlet 类public class Refresh extends HttpServlet {   // 处理 GET 方法请求的方法  public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 设置刷新自动加载时间为 5 秒      response.setIntHeader("Refresh", 5);       // 设置响应内容类型      response.setContentType("text/html");       // Get current time      Calendar calendar = new GregorianCalendar();      String am_pm;      int hour = calendar.get(Calendar.HOUR);      int minute = calendar.get(Calendar.MINUTE);      int second = calendar.get(Calendar.SECOND);      if(calendar.get(Calendar.AM_PM) == 0)        am_pm = "AM";      else        am_pm = "PM";       String CT = hour+":"+ minute +":"+ second +" "+ am_pm;          PrintWriter out = response.getWriter();      String title = "自动刷新 Header 设置";      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +        "<html>
" +        "<head><title>" + title + "</title></head>
"+        "<body bgcolor="#f0f0f0">
" +        "<h1 align="center">" + title + "</h1>
" +        "<p>当前时间是:" + CT + "</p>
");  }  // 处理 POST 方法请求的方法  public void doPost(HttpServletRequest request,                     HttpServletResponse response)      throws ServletException, IOException {     doGet(request, response);  }}

现在,调用上面的 Servlet,每隔 5 秒会显示当前系统时间。只要运行 Servlet 并稍等片刻,即可看到如下的结果:

自动刷新 Header 设置

当前时间是:9:44:50 PM

Servlet HTTP 状态码

HTTP 请求和 HTTP 响应消息的格式是类似的,结构如下:

  • 初始状态行 + 回车换行符(回车+换行)
  • 零个或多个标题行+回车换行符
  • 一个空白行,即回车换行符
  • 一个可选的消息主体,比如文件、查询数据或查询输出

例如,服务器的响应头如下所示:

HTTP/1.1 200 OKContent-Type: text/htmlHeader2: ......HeaderN: ...  (Blank Line)<!doctype ...><html><head>...</head><body>...</body></html>

状态行包括 HTTP 版本(在本例中为 HTTP/1.1)、一个状态码(在本例中为 200)和一个对应于状态码的短消息(在本例中为 OK)。

以下是可能从 Web 服务器返回的 HTTP 状态码和相关的信息列表:

代码 消息 描述
100 Continue 只有请求的一部分已经被服务器接收,但只要它没有被拒绝,客户端应继续该请求。
101 Switching Protocols 服务器切换协议。
200 OK 请求成功。
201 Created 该请求是完整的,并创建一个新的资源。
202 Accepted 该请求被接受处理,但是该处理是不完整的。
203 Non-authoritative Information  
204 No Content  
205 Reset Content  
206 Partial Content  
300 Multiple Choices 链接列表。用户可以选择一个链接,进入到该位置。最多五个地址。
301 Moved Permanently 所请求的页面已经转移到一个新的 URL。
302 Found 所请求的页面已经临时转移到一个新的 URL。
303 See Other 所请求的页面可以在另一个不同的 URL 下被找到。
304 Not Modified  
305 Use Proxy  
306 Unused 在以前的版本中使用该代码。现在已不再使用它,但代码仍被保留。
307 Temporary Redirect 所请求的页面已经临时转移到一个新的 URL。
400Bad Request 服务器不理解请求。
401 Unauthorized 所请求的页面需要用户名和密码。
402 Payment Required 您还不能使用该代码。
403 Forbidden 禁止访问所请求的页面。
404 Not Found 服务器无法找到所请求的页面。.
405 Method Not Allowed 在请求中指定的方法是不允许的。
406 Not Acceptable 服务器只生成一个不被客户端接受的响应。
407 Proxy Authentication Required 在请求送达之前,您必须使用代理服务器的验证。
408 Request Timeout 请求需要的时间比服务器能够等待的时间长,超时。
409 Conflict 请求因为冲突无法完成。
410 Gone 所请求的页面不再可用。
411 Length Required "Content-Length" 未定义。服务器无法处理客户端发送的不带 Content-Length 的请求信息。
412 Precondition Failed 请求中给出的先决条件被服务器评估为 false。
413 Request Entity Too Large 服务器不接受该请求,因为请求实体过大。
414 Request-url Too Long 服务器不接受该请求,因为 URL 太长。当您转换一个 "post" 请求为一个带有长的查询信息的 "get" 请求时发生。
415 Unsupported Media Type 服务器不接受该请求,因为媒体类型不被支持。
417 Expectation Failed  
500Internal Server Error 未完成的请求。服务器遇到了一个意外的情况。
501 Not Implemented 未完成的请求。服务器不支持所需的功能。
502 Bad Gateway 未完成的请求。服务器从上游服务器收到无效响应。
503 Service Unavailable 未完成的请求。服务器暂时超载或死机。
504 Gateway Timeout 网关超时。
505 HTTP Version Not Supported 服务器不支持"HTTP协议"版本。

设置 HTTP 状态代码的方法

下面的方法可用于在 Servlet 程序中设置 HTTP 状态码。这些方法通过 HttpServletResponse 对象可用。

序号方法 & 描述
1public void setStatus ( int statusCode )
该方法设置一个任意的状态码。setStatus 方法接受一个 int(状态码)作为参数。如果您的反应包含了一个特殊的状态码和文档,请确保在使用 PrintWriter 实际返回任何内容之前调用 setStatus。
2public void sendRedirect(String url)
该方法生成一个 302 响应,连同一个带有新文档 URL 的 Location 头。
3public void sendError(int code, String message)
该方法发送一个状态码(通常为 404),连同一个在 HTML 文档内部自动格式化并发送到客户端的短消息。

HTTP 状态码实例

下面的例子把 407 错误代码发送到客户端浏览器,浏览器会显示 "Need authentication!!!" 消息。

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*;import java.util.*;// 扩展 HttpServlet 类public class showError extends HttpServlet {   // 处理 GET 方法请求的方法  public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 设置错误代码和原因      response.sendError(407, "Need authentication!!!" );  }  // 处理 POST 方法请求的方法  public void doPost(HttpServletRequest request,                     HttpServletResponse response)      throws ServletException, IOException {     doGet(request, response);  }}

现在,调用上面的 Servlet 将显示以下结果:

HTTP Status 407 - Need authentication!!!

type Status report

message Need authentication!!!

description The client must first authenticate itself with the proxy (Need authentication!!!).

Apache Tomcat/5.5.29

Servlet 编写过滤器

Servlet 过滤器是可用于 Servlet 编程的 Java 类,有以下目的:

  • 在客户端的请求访问后端资源之前,拦截这些请求。
  • 在服务器的响应发送回客户端之前,处理这些响应。

根据规范建议的各种类型的过滤器:

  • 身份验证过滤器(Authentication Filters)。
  • 数据压缩过滤器(Data compression Filters)。
  • 加密过滤器(Encryption Filters)。
  • 触发资源访问事件过滤器。
  • 图像转换过滤器(Image Conversion Filters)。
  • 日志记录和审核过滤器(Logging and Auditing Filters)。
  • MIME-TYPE 链过滤器(MIME-TYPE Chain Filters)。
  • 标记化过滤器(Tokenizing Filters)。
  • XSL/T 过滤器(XSL/T Filters),转换 XML 内容。

过滤器被部署在部署描述符文件 web.xml 中,然后映射到您的应用程序的部署描述符中的 Servlet 名称或 URL 模式。

当 Web 容器启动 Web 应用程序时,它会为您在部署描述符中声明的每一个过滤器创建一个实例。该过滤器执行的顺序是按它们在部署描述符中声明的顺序。

Servlet 过滤器方法

过滤器是一个实现了 javax.servlet.Filter 接口的 Java 类。javax.servlet.Filter 接口定义了三个方法:

序号方法 & 描述
1public void doFilter (ServletRequest, ServletResponse, FilterChain)
该方法在每次一个请求/响应对因客户端在链的末端请求资源而通过链传递时由容器调用。
2public void init(FilterConfig filterConfig)
该方法由 Web 容器调用,指示一个过滤器被放入服务。
3public void destroy()
该方法由 Web 容器调用,指示一个过滤器被取出服务。

FilterConfig 使用

Filter 的 init 方法中提供了一个 FilterConfig 对象。

如 web.xml 文件配置如下:

<filter>	<filter-name>LoginFilter</filter-name>	<filter-class>com.51coolma.test.LogFilter</filter-class>	<init-param>		<param-name>Site</param-name>		<param-value>51coolma在线教程</param-value>	</init-param>	</filter>

在 init 方法使用 FilterConfig 对象获取参数:

public void  init(FilterConfig config) throws ServletException {	// 获取初始化参数	String site = config.getInitParameter("Site"); 	// 输出初始化参数	System.out.println("网站名称: " + site); }

Servlet 过滤器实例

以下是 Servlet 过滤器的实例,将输出客户端的 IP 地址和当前的日期时间。本实例让您对 Servlet 过滤器有基本的了解,您可以使用相同的概念编写更复杂的过滤器应用程序:

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*;import java.util.*;// 实现 Filter 类public class LogFilter implements Filter  {   public void  init(FilterConfig config)                          throws ServletException{      // 获取初始化参数      String testParam = config.getInitParameter("test-param");       // 输出初始化参数      System.out.println("Test Param: " + testParam);    }   public void  doFilter(ServletRequest request,                  ServletResponse response,                 FilterChain chain)                  throws java.io.IOException, ServletException {      // 获取客户机的 IP 地址         String ipAddress = request.getRemoteAddr();      // 记录 IP 地址和当前时间戳      System.out.println("IP "+ ipAddress + ", Time "                                       + new Date().toString());      // 把请求传回过滤链      chain.doFilter(request,response);   }   public void destroy( ){      /* 在 Filter 实例被 Web 容器从服务移除之前调用 */   }}

这边使用前文提到的 DisplayHeader.java 为例子:

//导入必需的 java 库import java.io.IOException;import java.io.PrintWriter;import java.util.Enumeration;import javax.servlet.ServletException;import javax.servlet.annotation.WebServlet;import javax.servlet.http.HttpServlet;import javax.servlet.http.HttpServletRequest;import javax.servlet.http.HttpServletResponse;@WebServlet("/DisplayHeader")//扩展 HttpServlet 类public class DisplayHeader extends HttpServlet {	// 处理 GET 方法请求的方法	public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException	{		// 设置响应内容类型		response.setContentType("text/html;charset=UTF-8");		PrintWriter out = response.getWriter();		String title = "HTTP Header 请求实例 - 51coolma教程实例";		String docType =			"<!DOCTYPE html> 
";			out.println(docType +			"<html>
" +			"<head><meta charset="utf-8"><title>" + title + "</title></head>
"+			"<body bgcolor="#f0f0f0">
" +			"<h1 align="center">" + title + "</h1>
" +			"<table width="100%" border="1" align="center">
" +			"<tr bgcolor="#949494">
" +			"<th>Header 名称</th><th>Header 值</th>
"+			"</tr>
");		Enumeration headerNames = request.getHeaderNames();		while(headerNames.hasMoreElements()) {			String paramName = (String)headerNames.nextElement();			out.print("<tr><td>" + paramName + "</td>
");			String paramValue = request.getHeader(paramName);			out.println("<td> " + paramValue + "</td></tr>
");		}		out.println("</table>
</body></html>");	}	// 处理 POST 方法请求的方法	public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {		doGet(request, response);	}}

Web.xml 中的 Servlet 过滤器映射(Servlet Filter Mapping)

定义过滤器,然后映射到一个 URL 或 Servlet,这与定义 Servlet,然后映射到一个 URL 模式方式大致相同。在部署描述符文件 web.xml 中为 filter 标签创建下面的条目:

<?xml version="1.0" encoding="UTF-8"?>  <web-app>  <filter>  <filter-name>LogFilter</filter-name>  <filter-class>com.51coolma.test.LogFilter</filter-class>  <init-param>    <param-name>Site</param-name>    <param-value>51coolma在线教程</param-value>  </init-param></filter><filter-mapping>  <filter-name>LogFilter</filter-name>  <url-pattern>/*</url-pattern></filter-mapping><servlet>    <!-- 类名 -->    <servlet-name>DisplayHeader</servlet-name>    <!-- 所在的包 -->    <servlet-class>com.51coolma.test.DisplayHeader</servlet-class>  </servlet>  <servlet-mapping>    <servlet-name>DisplayHeader</servlet-name>    <!-- 访问的网址 -->    <url-pattern>/TomcatTest/DisplayHeader</url-pattern>  </servlet-mapping>  </web-app>  

上述过滤器适用于所有的 Servlet,因为我们在配置中指定 /* 。如果您只想在少数的 Servlet 上应用过滤器,您可以指定一个特定的 Servlet 路径。

现在试着以常用的方式调用任何 Servlet,您将会看到在 Web 服务器中生成的日志。您也可以使用 Log4J 记录器来把上面的日志记录到一个单独的文件中。

使用多个过滤器

Web 应用程序可以根据特定的目的定义若干个不同的过滤器。假设您定义了两个过滤器 AuthenFilterLogFilter。您需要创建一个如下所述的不同的映射,其余的处理与上述所讲解的大致相同:

<filter>   <filter-name>LogFilter</filter-name>   <filter-class>com.51coolma.test.LogFilter</filter-class>   <init-param>	  <param-name>test-param</param-name>	  <param-value>Initialization Paramter</param-value>   </init-param></filter><filter>   <filter-name>AuthenFilter</filter-name>   <filter-class>com.51coolma.test.AuthenFilter</filter-class>   <init-param>	  <param-name>test-param</param-name>	  <param-value>Initialization Paramter</param-value>   </init-param></filter><filter-mapping>   <filter-name>LogFilter</filter-name>   <url-pattern>/*</url-pattern></filter-mapping><filter-mapping>   <filter-name>AuthenFilter</filter-name>   <url-pattern>/*</url-pattern></filter-mapping>

过滤器的应用顺序

web.xml 中的 filter-mapping 元素的顺序决定了 Web 容器应用过滤器到 Servlet 的顺序。若要反转过滤器的顺序,您只需要在 web.xml 文件中反转 filter-mapping 元素即可。

例如,上面的实例将先应用 LogFilter,然后再应用 AuthenFilter,但是下面的实例将颠倒这个顺序:

<filter-mapping>   <filter-name>AuthenFilter</filter-name>   <url-pattern>/*</url-pattern></filter-mapping><filter-mapping>   <filter-name>LogFilter</filter-name>   <url-pattern>/*</url-pattern></filter-mapping>

web.xml配置各节点说明

<filter>指定一个过滤器。
<filter-name>用于为过滤器指定一个名字,该元素的内容不能为空。
<filter-class>元素用于指定过滤器的完整的限定类名。
<init-param>元素用于为过滤器指定初始化参数,它的子元素<param-name>指定参数的名字,<param-value>指定参数的值。
在过滤器中,可以使用FilterConfig接口对象来访问初始化参数。
<filter-mapping>元素用于设置一个 Filter 所负责拦截的资源。一个Filter拦截的资源可通过两种方式来指定:Servlet 名称和资源访问的请求路径
<filter-name>子元素用于设置filter的注册名称。该值必须是在<filter>元素中声明过的过滤器的名字
<url-pattern>设置 filter 所拦截的请求路径(过滤器关联的URL样式)
<servlet-name>指定过滤器所拦截的Servlet名称。
<dispatcher>指定过滤器所拦截的资源被 Servlet 容器调用的方式,可以是REQUEST,INCLUDE,FORWARD和ERROR之一,默认REQUEST。用户可以设置多个<dispatcher>子元素用来指定 Filter 对资源的多种调用方式进行拦截。
<dispatcher>子元素可以设置的值及其意义
REQUEST:当用户直接访问页面时,Web容器将会调用过滤器。如果目标资源是通过RequestDispatcher的include()或forward()方法访问时,那么该过滤器就不会被调用。
INCLUDE:如果目标资源是通过RequestDispatcher的include()方法访问时,那么该过滤器将被调用。除此之外,该过滤器不会被调用。
FORWARD:如果目标资源是通过RequestDispatcher的forward()方法访问时,那么该过滤器将被调用,除此之外,该过滤器不会被调用。
ERROR:如果目标资源是通过声明式异常处理机制调用时,那么该过滤器将被调用。除此之外,过滤器不会被调用。

Servlet 异常处理

当一个 Servlet 抛出一个异常时,Web 容器在使用了 exception-type 元素的 web.xml 中搜索与抛出异常类型相匹配的配置。

您必须在 web.xml 中使用 error-page 元素来指定对特定异常 或 HTTP 状态码 作出相应的 Servlet 调用。

web.xml 配置

假设,有一个 ErrorHandler 的 Servlet 在任何已定义的异常或错误出现时被调用。以下将是在 web.xml 中创建的项。

<!-- servlet 定义 --><servlet>        <servlet-name>ErrorHandler</servlet-name>        <servlet-class>ErrorHandler</servlet-class></servlet><!-- servlet 映射 --><servlet-mapping>        <servlet-name>ErrorHandler</servlet-name>        <url-pattern>/ErrorHandler</url-pattern></servlet-mapping><!-- error-code 相关的错误页面 --><error-page>    <error-code>404</error-code>    <location>/ErrorHandler</location></error-page><error-page>    <error-code>403</error-code>    <location>/ErrorHandler</location></error-page><!-- exception-type 相关的错误页面 --><error-page>    <exception-type>          javax.servlet.ServletException    </exception-type >    <location>/ErrorHandler</location></error-page><error-page>    <exception-type>java.io.IOException</exception-type >    <location>/ErrorHandler</location></error-page>

如果您想对所有的异常有一个通用的错误处理程序,那么应该定义下面的 error-page,而不是为每个异常定义单独的 error-page 元素:

<error-page>    <exception-type>java.lang.Throwable</exception-type >    <location>/ErrorHandler</location></error-page>

以下是关于上面的 web.xml 异常处理要注意的点:

  • Servelt ErrorHandler 与其他的 Servelt 的定义方式一样,且在 web.xml 中进行配置。
  • 如果有错误状态代码出现,不管为 404(Not Found 未找到)或 403(Forbidden 禁止),则会调用 ErrorHandler 的 Servlet。
  • 如果 Web 应用程序抛出 ServletExceptionIOException,那么 Web 容器会调用 ErrorHandler 的 Servlet。
  • 您可以定义不同的错误处理程序来处理不同类型的错误或异常。上面的实例是非常通用的,希望您能通过实例理解基本的概念。

请求属性 - 错误/异常

以下是错误处理的 Servlet 可以访问的请求属性列表,用来分析错误/异常的性质。

序号属性 & 描述
1javax.servlet.error.status_code
该属性给出状态码,状态码可被存储,并在存储为 java.lang.Integer 数据类型后可被分析。
2javax.servlet.error.exception_type
该属性给出异常类型的信息,异常类型可被存储,并在存储为 java.lang.Class 数据类型后可被分析。
3javax.servlet.error.message
该属性给出确切错误消息的信息,信息可被存储,并在存储为 java.lang.String 数据类型后可被分析。
4javax.servlet.error.request_uri
该属性给出有关 URL 调用 Servlet 的信息,信息可被存储,并在存储为 java.lang.String 数据类型后可被分析。
5javax.servlet.error.exception
该属性给出异常产生的信息,信息可被存储,并在存储为 java.lang.Throwable 数据类型后可被分析。
6javax.servlet.error.servlet_name
该属性给出 Servlet 的名称,名称可被存储,并在存储为 java.lang.String 数据类型后可被分析。

Servlet 错误处理程序实例

以下是 Servlet 实例,将应对任何您所定义的错误或异常发生时的错误处理程序。

本实例让您对 Servlet 中的异常处理有基本的了解,您可以使用相同的概念编写更复杂的异常处理应用程序:

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*;import java.util.*;// 扩展 HttpServlet 类public class ErrorHandler extends HttpServlet {   // 处理 GET 方法请求的方法  public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 分析 Servlet 异常             Throwable throwable = (Throwable)      request.getAttribute("javax.servlet.error.exception");      Integer statusCode = (Integer)      request.getAttribute("javax.servlet.error.status_code");      String servletName = (String)      request.getAttribute("javax.servlet.error.servlet_name");      if (servletName == null){         servletName = "Unknown";      }      String requestUri = (String)      request.getAttribute("javax.servlet.error.request_uri");      if (requestUri == null){         requestUri = "Unknown";      }      // 设置响应内容类型      response.setContentType("text/html");       PrintWriter out = response.getWriter();     String title = "Error/Exception Information";      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +        "<html>
" +        "<head><title>" + title + "</title></head>
" +        "<body bgcolor="#f0f0f0">
");      if (throwable == null && statusCode == null){         out.println("<h2>Error information is missing</h2>");         out.println("Please return to the <a href="" +             response.encodeURL("http://localhost:8080/") +             "">Home Page</a>.");      }else if (statusCode != null){         out.println("The status code : " + statusCode);      }else{         out.println("<h2 class="tutheader">Error information</h2>");         out.println("Servlet Name : " + servletName +                              "</br></br>");         out.println("Exception Type : " +                              throwable.getClass( ).getName( ) +                              "</br></br>");         out.println("The request URI: " + requestUri +                              "<br><br>");         out.println("The exception message: " +                                  throwable.getMessage( ));      }      out.println("</body>");      out.println("</html>");  }  // 处理 POST 方法请求的方法  public void doPost(HttpServletRequest request,                     HttpServletResponse response)      throws ServletException, IOException {     doGet(request, response);  }}

以通常的方式编译 ErrorHandler.java,把您的类文件放入<Tomcat-installation-directory>/webapps/ROOT/WEB-INF/classes 中。

让我们在 web.xml 文件中添加如下配置来处理异常:

<servlet>        <servlet-name>ErrorHandler</servlet-name>        <servlet-class>ErrorHandler</servlet-class></servlet><!-- servlet mappings --><servlet-mapping>        <servlet-name>ErrorHandler</servlet-name>        <url-pattern>/ErrorHandler</url-pattern></servlet-mapping><error-page>    <error-code>404</error-code>    <location>/ErrorHandler</location></error-page><error-page>    <exception-type>java.lang.Throwable</exception-type >    <location>/ErrorHandler</location></error-page>

现在,尝试使用一个会产生异常的 Servlet,或者输入一个错误的 URL,这将触发 Web 容器调用 ErrorHandler 的 Servlet,并显示适当的消息。例如,如果您输入了一个错误的 URL,那么它将显示下面的结果:

The status code : 404

上面的代码在某些 Web 浏览器中可能无法正常工作。因此,请尽量尝试使用 Mozilla 和 Safari 浏览器,在这两种浏览器中它们应该能够正常工作。


Servlet Cookies 处理

Cookies 是存储在客户端计算机上的文本文件,并保留了各种跟踪信息。Java Servlet 显然支持 HTTP Cookies。

识别返回用户包括三个步骤:

  • 服务器脚本向浏览器发送一组 Cookies。例如:姓名、年龄或识别号码等。
  • 浏览器将这些信息存储在本地计算机上,以备将来使用。
  • 当下一次浏览器向 Web 服务器发送任何请求时,浏览器会把这些 Cookies 信息发送到服务器,服务器将使用这些信息来识别用户。

本章将向您讲解如何设置或重置 Cookies,如何访问它们,以及如何将它们删除。

Servlet Cookie 处理需要对中文进行编码与解码,方法如下:

String   str   =   java.net.URLEncoder.encode("中文");            //编码String   str   =   java.net.URLDecoder.decode("编码后的字符串");   // 解码

Cookie 剖析

Cookies 通常设置在 HTTP 头信息中(虽然 JavaScript 也可以直接在浏览器上设置一个 Cookie)。设置 Cookie 的 Servlet 会发送如下的头信息:

HTTP/1.1 200 OKDate: Fri, 04 Feb 2000 21:03:38 GMTServer: Apache/1.3.9 (UNIX) PHP/4.0b3Set-Cookie: name=xyz; expires=Friday, 04-Feb-07 22:03:38 GMT;                  path=/; domain=51coolma.cnConnection: closeContent-Type: text/html

正如您所看到的,Set-Cookie 头包含了一个名称值对、一个 GMT 日期、一个路径和一个域。名称和值会被 URL 编码。expires 字段是一个指令,告诉浏览器在给定的时间和日期之后"忘记"该 Cookie。

如果浏览器被配置为存储 Cookies,它将会保留此信息直到到期日期。如果用户的浏览器指向任何匹配该 Cookie 的路径和域的页面,它会重新发送 Cookie 到服务器。浏览器的头信息可能如下所示:

GET / HTTP/1.0Connection: Keep-AliveUser-Agent: Mozilla/4.6 (X11; I; Linux 2.2.6-15apmac ppc)Host: zink.demon.co.uk:1126Accept: image/gif, */*Accept-Encoding: gzipAccept-Language: enAccept-Charset: iso-8859-1,*,utf-8Cookie: name=xyz

Servlet 就能够通过请求方法 request.getCookies() 访问 Cookie,该方法将返回一个 Cookie 对象的数组。

Servlet Cookies 方法

以下是在 Servlet 中操作 Cookies 时可使用的有用的方法列表。

序号方法 & 描述
1public void setDomain(String pattern)
该方法设置 cookie 适用的域,例如 51coolma.cn。
2public String getDomain()
该方法获取 cookie 适用的域,例如 51coolma.cn。
3public void setMaxAge(int expiry)
该方法设置 cookie 过期的时间(以秒为单位)。如果不这样设置,cookie 只会在当前 session 会话中持续有效。
4public int getMaxAge()
该方法返回 cookie 的最大生存周期(以秒为单位),默认情况下,-1 表示 cookie 将持续下去,直到浏览器关闭。
5public String getName()
该方法返回 cookie 的名称。名称在创建后不能改变。
6public void setValue(String newValue)
该方法设置与 cookie 关联的值。
7public String getValue()
该方法获取与 cookie 关联的值。
8public void setPath(String uri)
该方法设置 cookie 适用的路径。如果您不指定路径,与当前页面相同目录下的(包括子目录下的)所有 URL 都会返回 cookie。
9public String getPath()
该方法获取 cookie 适用的路径。
10public void setSecure(boolean flag)
该方法设置布尔值,表示 cookie 是否应该只在加密的(即 SSL)连接上发送。
11public void setComment(String purpose)
该方法规定了描述 cookie 目的的注释。该注释在浏览器向用户呈现 cookie 时非常有用。
12public String getComment()
该方法返回了描述 cookie 目的的注释,如果 cookie 没有注释则返回 null。

通过 Servlet 设置 Cookies

通过 Servlet 设置 Cookies 包括三个步骤:

(1) 创建一个 Cookie 对象:您可以调用带有 cookie 名称和 cookie 值的 Cookie 构造函数,cookie 名称和 cookie 值都是字符串。

Cookie cookie = new Cookie("key","value");

请记住,无论是名字还是值,都不应该包含空格或以下任何字符:

[ ] ( ) = , " / ? @ : ;

(2) 设置最大生存周期:您可以使用 setMaxAge 方法来指定 cookie 能够保持有效的时间(以秒为单位)。下面将设置一个最长有效期为 24 小时的 cookie。

cookie.setMaxAge(60*60*24); 

(3) 发送 Cookie 到 HTTP 响应头:您可以使用 response.addCookie 来添加 HTTP 响应头中的 Cookies,如下所示:

response.addCookie(cookie);

实例

让我们修改我们的 表单数据实例,为名字和姓氏设置 Cookies。

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*; // 扩展 HttpServlet 类public class HelloForm extends HttpServlet {   public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 为名字和姓氏创建 Cookies            Cookie firstName = new Cookie("first_name",                      request.getParameter("first_name"));      Cookie lastName = new Cookie("last_name",                      request.getParameter("last_name"));      // 为两个 Cookies 设置过期日期为 24 小时后      firstName.setMaxAge(60*60*24);       lastName.setMaxAge(60*60*24);       // 在响应头中添加两个 Cookies      response.addCookie( firstName );      response.addCookie( lastName );      // 设置响应内容类型      response.setContentType("text/html");       PrintWriter out = response.getWriter();      String title = "设置 Cookies 实例";      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +                "<html>
" +                "<head><title>" + title + "</title></head>
" +                "<body bgcolor="#f0f0f0">
" +                "<h1 align="center">" + title + "</h1>
" +                "<ul>
" +                "  <li><b>名字</b>:"                + request.getParameter("first_name") + "
" +                "  <li><b>姓氏</b>:"                + request.getParameter("last_name") + "
" +                "</ul>
" +                "</body></html>");  }}

编译上面的 Servlet HelloForm,并在 web.xml 文件中创建适当的条目,最后尝试下面的 HTML 页面来调用 Servlet。

 <html><body><form action="HelloForm" method="GET">名字:<input type="text" name="first_name"><br />姓氏:<input type="text" name="last_name" /><input type="submit" value="提交" /></form></body></html>

保存上面的 HTML 内容到文件 hello.htm 中,并把它放在 <Tomcat-installation-directory>/webapps/ROOT 目录中。当您访问 http://localhost:8080/Hello.htm 时,上面表单的实际输出如下所示:


尝试输入名字和姓氏,然后点击"提交"按钮,名字和姓氏将显示在屏幕上,同时会设置 firstName 和 lastName 这两个 Cookies,当下次您按下提交按钮时,会将这两个 Cookies 传回到服务器。

下一节会讲解如何在 Web 应用程序中访问这些 Cookies。

通过 Servlet 读取 Cookies

要读取 Cookies,您需要通过调用 HttpServletRequestgetCookies( ) 方法创建一个 javax.servlet.http.Cookie 对象的数组。然后循环遍历数组,并使用 getName() 和 getValue() 方法来访问每个 cookie 和关联的值。

实例

让我们读取上面的实例中设置的 Cookies

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*; // 扩展 HttpServlet 类public class ReadCookies extends HttpServlet {   public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      Cookie cookie = null;    Cookie[] cookies = null;      // 获取与该域相关的 Cookies 的数组      cookies = request.getCookies();          // 设置响应内容类型      response.setContentType("text/html");       PrintWriter out = response.getWriter();      String title = "Reading Cookies Example";      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +                "<html>
" +                "<head><title>" + title + "</title></head>
" +                "<body bgcolor="#f0f0f0">
" );      if( cookies != null ){         out.println("<h2>查找 Cookies 名称和值</h2>");         for (int i = 0; i < cookies.length; i++){             cookie = cookies[i];             out.print("名称:" + cookie.getName( ) + ",");             out.print("值:" + cookie.getValue( )+" <br/>");         }      }else{          out.println(            "<h2 class="tutheader">未找到 Cookies</h2>");      }      out.println("</body>");      out.println("</html>");   }}

编译上面的 Servlet ReadCookies,并在 web.xml 文件中创建适当的条目。如果您已经设置了 first_name cookie 为 "John",last_name cookie 为 "Player" ,尝试运行 http://localhost:8080/ReadCookies,将显示如下结果:

查找 Cookies 名称和值

名称:first_name,值:John
名称:last_name,值:Player

通过 Servlet 删除 Cookies

删除 Cookies 是非常简单的。如果您想删除一个 cookie,那么您只需要按照以下三个步骤进行:

  • 读取一个现有的 cookie,并把它存储在 Cookie 对象中。
  • 使用 setMaxAge() 方法设置 cookie 的年龄为零,来删除现有的 cookie。
  • 把这个 cookie 添加到响应头。

实例

下面的例子将删除现有的名为 "first_name" 的 cookie,当您下次运行 ReadCookies 的 Servlet 时,它会返回 first_name 为空值。

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*; // 扩展 HttpServlet 类public class DeleteCookies extends HttpServlet {   public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      Cookie cookie = null;    Cookie[] cookies = null;      // 获取与该域相关的 Cookies 的数组      cookies = request.getCookies();          // 设置响应内容类型      response.setContentType("text/html");       PrintWriter out = response.getWriter();      String title = "Delete Cookies Example";      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +                "<html>
" +                "<head><title>" + title + "</title></head>
" +                "<body bgcolor="#f0f0f0">
" );       if( cookies != null ){         out.println("<h2>Cookies 名称和值</h2>");         for (int i = 0; i < cookies.length; i++){             cookie = cookies[i];             if((cookie.getName( )).compareTo("first_name") == 0 ){                  cookie.setMaxAge(0);                  response.addCookie(cookie);                  out.print("已删除的 cookie:" +                                cookie.getName( ) + "<br/>");            }            out.print("名称:" + cookie.getName( ) + ",");            out.print("值:" + cookie.getValue( )+" <br/>");         }      }else{          out.println(            "<h2 class="tutheader">No cookies founds</h2>");      }      out.println("</body>");      out.println("</html>");   }}

编译上面的 Servlet DeleteCookies,并在 web.xml 文件中创建适当的条目。现在运行 http://localhost:8080/DeleteCookies,将显示如下结果:

Cookies 名称和值

已删除的 cookie:first_name
名称:first_name,值:John
名称:last_name,值:Player

现在尝试运行 http://localhost:8080/ReadCookies,它将只显示一个 cookie,如下所示:

查找 Cookies 名称和值

名称:last_name,值:Player

您可以手动在 Internet Explorer 中删除 Cookies。在"工具"菜单,选择"Internet 选项"。如果要删除所有的 Cookies,请按"删除 Cookies"。

Servlet Session 跟踪

HTTP 是一种"无状态"协议,这意味着每次客户端检索网页时,客户端打开一个单独的连接到 Web 服务器,服务器会自动不保留之前客户端请求的任何记录。

但是仍然有以下三种方式来维持 Web 客户端和 Web 服务器之间的 session 会话:

Cookies

一个 Web 服务器可以分配一个唯一的 session 会话 ID 作为每个 Web 客户端的 cookie,对于客户端的后续请求可以使用接收到的 cookie 来识别。

这可能不是一个有效的方法,因为很多浏览器不支持 cookie,所以我们建议不要使用这种方式来维持 session 会话。

隐藏的表单字段

一个 Web 服务器可以发送一个隐藏的 HTML 表单字段,以及一个唯一的 session 会话 ID,如下所示:

<input type="hidden" name="sessionid" value="12345">

该条目意味着,当表单被提交时,指定的名称和值会被自动包含在 GET 或 POST 数据中。每次当 Web 浏览器发送回请求时,session_id 值可以用于保持不同的 Web 浏览器的跟踪。

这可能是一种保持 session 会话跟踪的有效方式,但是点击常规的超文本链接(<A HREF...>)不会导致表单提交,因此隐藏的表单字段也不支持常规的 session 会话跟踪。

URL 重写

您可以在每个 URL 末尾追加一些额外的数据来标识 session 会话,服务器会把该 session 会话标识符与已存储的有关 session 会话的数据相关联。

例如,http://51coolma.cn/file.htm;sessionid=12345,session 会话标识符被附加为 sessionid=12345,标识符可被 Web 服务器访问以识别客户端。

URL 重写是一种更好的维持 session 会话的方式,它在浏览器不支持 cookie 时能够很好地工作,但是它的缺点是会动态生成每个 URL 来为页面分配一个 session 会话 ID,即使是在很简单的静态 HTML 页面中也会如此。

HttpSession 对象

除了上述的三种方式,Servlet 还提供了 HttpSession 接口,该接口提供了一种跨多个页面请求或访问网站时识别用户以及存储有关用户信息的方式。

Servlet 容器使用这个接口来创建一个 HTTP 客户端和 HTTP 服务器之间的 session 会话。会话持续一个指定的时间段,跨多个连接或页面请求。

您会通过调用 HttpServletRequest 的公共方法 getSession() 来获取 HttpSession 对象,如下所示:

HttpSession session = request.getSession();

你需要在向客户端发送任何文档内容之前调用 request.getSession()。下面总结了 HttpSession 对象中可用的几个重要的方法:

序号方法 & 描述
1public Object getAttribute(String name)
该方法返回在该 session 会话中具有指定名称的对象,如果没有指定名称的对象,则返回 null。
2public Enumeration getAttributeNames()
该方法返回 String 对象的枚举,String 对象包含所有绑定到该 session 会话的对象的名称。
3public long getCreationTime()
该方法返回该 session 会话被创建的时间,自格林尼治标准时间 1970 年 1 月 1 日午夜算起,以毫秒为单位。
4public String getId()
该方法返回一个包含分配给该 session 会话的唯一标识符的字符串。
5public long getLastAccessedTime()
该方法返回客户端最后一次发送与该 session 会话相关的请求的时间自格林尼治标准时间 1970 年 1 月 1 日午夜算起,以毫秒为单位。
6public int getMaxInactiveInterval()
该方法返回 Servlet 容器在客户端访问时保持 session 会话打开的最大时间间隔,以秒为单位。
7public void invalidate()
该方法指示该 session 会话无效,并解除绑定到它上面的任何对象。
8public boolean isNew(
如果客户端还不知道该 session 会话,或者如果客户选择不参入该 session 会话,则该方法返回 true。
9public void removeAttribute(String name)
该方法将从该 session 会话移除指定名称的对象。
10public void setAttribute(String name, Object value)
该方法使用指定的名称绑定一个对象到该 session 会话。
11public void setMaxInactiveInterval(int interval)
该方法在 Servlet 容器指示该 session 会话无效之前,指定客户端请求之间的时间,以秒为单位。

Session 跟踪实例

本实例说明了如何使用 HttpSession 对象获取 session 会话创建时间和最后访问时间。如果不存在 session 会话,我们将通过请求创建一个新的 session 会话。

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*;import java.util.*; // 扩展 HttpServlet 类public class SessionTrack extends HttpServlet {   public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 如果不存在 session 会话,则创建一个 session 对象      HttpSession session = request.getSession(true);      // 获取 session 创建时间      Date createTime = new Date(session.getCreationTime());      // 获取该网页的最后一次访问时间      Date lastAccessTime =                         new Date(session.getLastAccessedTime());      String title = "欢迎回到我的网站";      Integer visitCount = new Integer(0);      String visitCountKey = new String("visitCount");      String userIDKey = new String("userID");      String userID = new String("ABCD");      // 检查网页上是否有新的访问者      if (session.isNew()){         title = "欢迎来到我的网站";         session.setAttribute(userIDKey, userID);      } else {         visitCount = (Integer)session.getAttribute(visitCountKey);         visitCount = visitCount + 1;         userID = (String)session.getAttribute(userIDKey);      }      session.setAttribute(visitCountKey,  visitCount);      // 设置响应内容类型      response.setContentType("text/html");      PrintWriter out = response.getWriter();      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +                "<html>
" +                "<head><title>" + title + "</title></head>
" +                "<body bgcolor="#f0f0f0">
" +                "<h1 align="center">" + title + "</h1>
" +                 "<h2 align="center">Session 信息</h2>
" +                "<table border="1" align="center">
" +                "<tr bgcolor="#949494">
" +                "  <th>Session 信息</th><th>值</th></tr>
" +                "<tr>
" +                "  <td>id</td>
" +                "  <td>" + session.getId() + "</td></tr>
" +                "<tr>
" +                "  <td>Creation Time</td>
" +                "  <td>" + createTime +                 "  </td></tr>
" +                "<tr>
" +                "  <td>Time of Last Access</td>
" +                "  <td>" + lastAccessTime +                 "  </td></tr>
" +                "<tr>
" +                "  <td>User ID</td>
" +                "  <td>" + userID +                 "  </td></tr>
" +                "<tr>
" +                "  <td>Number of visits</td>
" +                "  <td>" + visitCount + "</td></tr>
" +                "</table>
" +                "</body></html>");  }}

编译上面的 Servlet SessionTrack,并在 web.xml 文件中创建适当的条目。在浏览器地址栏输入 http://localhost:8080/SessionTrack,当您第一次运行时将显示如下结果:

欢迎来到我的网站

Session 信息

Session 信息
id 0AE3EC93FF44E3C525B4351B77ABB2D5
Creation Time Tue Jun 08 17:26:40 GMT+04:00 2014
Time of Last Access Tue Jun 08 17:26:40 GMT+04:00 2014
User ID ABCD
Number of visits 0

再次尝试运行相同的 Servlet,它将显示如下结果:

欢迎回到我的网站

Session 信息

Session 信息
id 0AE3EC93FF44E3C525B4351B77ABB2D5
Creation Time Tue Jun 08 17:26:40 GMT+04:00 2014
Time of Last Access Tue Jun 08 17:26:40 GMT+04:00 2014
User ID ABCD
Number of visits 1

删除 Session 会话数据

当您完成了一个用户的 session 会话数据,您有以下几种选择:

  • 移除一个特定的属性:您可以调用 public void removeAttribute(String name) 方法来删除与特定的键相关联的值。 to delete the value associated with a particular key.
  • 删除整个 session 会话:您可以调用 public void invalidate() 方法来丢弃整个 session 会话。
  • 设置 session 会话过期时间:您可以调用 public void setMaxInactiveInterval(int interval) 方法来单独设置 session 会话超时。
  • 注销用户:如果使用的是支持 servlet 2.4 的服务器,您可以调用 logout 来注销 Web 服务器的客户端,并把属于所有用户的所有 session 会话设置为无效。
  • web.xml 配置:如果您使用的是 Tomcat,除了上述方法,您还可以在 web.xml 文件中配置 session 会话超时,如下所示:
  <session-config>    <session-timeout>15</session-timeout>  </session-config>

上面实例中的超时时间是以分钟为单位,将覆盖 Tomcat 中默认的 30 分钟超时时间。

在一个 Servlet 中的 getMaxInactiveInterval() 方法会返回 session 会话的超时时间,以秒为单位。所以,如果在 web.xml 中配置 session 会话超时时间为 15 分钟,那么 getMaxInactiveInterval() 会返回 900。

ServletContext是一个全局的储存信息的空间,服务器开始就存在,服务器关闭才释放。为了方便大家理解,我们将ServletContext和Cookie、Session做一个简单对比,如下图:


1


我们可以把ServletContext当成一个公用的空间,可以被所有的客户访问,如上图,A、B、C三个客户端都可以访问。

WEB容器在启动时,它会为每个Web应用程序都创建一个对应的ServletContext,它代表当前Web应用,并且它被所有客户端共享。


由于一个WEB应用中的所有Servlet共享同一个ServletContext对象,因此Servlet对象之间可以通过ServletContext对象来实现通讯。ServletContext对象通常也被称之为context域对象。公共聊天室就会用到它。

当web应用关闭、Tomcat关闭或者Web应用reload的时候,ServletContext对象会被销毁


ServletContext使用方法


1、ServletContext对象如何得到

this.getServletContext(); this.getServletConfig().getServletContext();

2、你可以把它想象成一张表,这个和Session非常相似:每一行就是一个属性,如下:

名字(String)值(Object)


添加属性:setAttribute(String name, Object obj);

得到值:getAttribute(String name),这个方法返回Object

删除属性:removeAttribute(String name)


3、生命周期 

ServletContext中的属性的生命周期从创建开始,到服务器关闭结束。


一个快速入门的案例: 

我们创建Servlet1和Servlet2,分别用于在ServletContext中创建和读取属性: 


Servlet1的doGet方法为:

public void doGet(HttpServletRequest request, HttpServletResponse response)        throws ServletException, IOException {    response.setContentType("text/html;charset=utf-8");    PrintWriter out = response.getWriter();    // 获取ServletContext对象的引用    // 第一种方法    ServletContext servletContext = this.getServletContext();    // 第二种方法    // ServletContext servletContext2 = this.getServletConfig().getServletContext();    servletContext.setAttribute("name", "小明");    out.println("将 name=小明  写入了ServletContext");}

Servlet2的doGet方法为:

public void doGet(HttpServletRequest request, HttpServletResponse response)        throws ServletException, IOException {    response.setContentType("text/html;charset=utf-8");    PrintWriter out = response.getWriter();    // 取出ServletContext的某个属性    //1.首先获取到ServletContext    ServletContext servletContext = this.getServletContext();    //2.取出属性    String name = (String)servletContext.getAttribute("name");    out.println("name="+name);}

以此访问Servlet1和Servlet2,我们可以分别看到输出如下:


Servletcontext 对象


Servletcontext 对象


粗看之下,这个运行结果和Session,Cookie的应用似乎没什么区别,但事实上则完全不一样的。只要不关闭Tomcat或者reload该应用,当我们关闭当前的浏览器,或者是换一个浏览器,比如从360浏览器换到了IE浏览器再次访问Servlet2,我们依然可以看到这个结果!这就是和和Session,Cookie最大的不同了。之所以会造成这种不同,是因为ServletContext存在于服务器内存中的一个公共空间,它可以供所有的用户客户端访问。


ServletContext应用


1、多个Servlet可以通过ServletContext对象来实现数据间的共享

类似于Session,通过ServletContext对象我们也可以实现数据共享,但值得注意的是,Session是只能在一个客户端中共享数据,而ServletContext中的数据是在所有客户端中都可以实现数据共享的。


2、实现Servlet的请求转发

之前我们学过的请求转发是通过request对象的: 

request.getRequestDispatcher("/url").forward(request, response);

这里要说明的是,ServletContext也可以实现请求转发: 

this.getServletContext().getRequestDispatcher("/url").forward(request, response); 

这两个转发效果是一样的。


3、获取Web应用的初始化参数

我们可以用<init-param>标签为servlet配置初始化参数,然后使用ServletConfig对象获取这些参数,假如有如下的MyServlet,它的配置为:

<servlet>      <servlet-name>MyServlet</servlet-name>      <servlet-class>com.gavin.servlet.MyServlet</servlet-class>      <init-param>          <param-name>encoding</param-name>          <param-value>utf-8</param-value>      </init-param>  </servlet>  

可以看到它配置了一个初始化参数:encoding=utf-8,那么我们在MyServlet的源代码中需要这样去得到这个参数:

String encoding = this.getServletConfig().getInitParameter("encoding");

注意,上述的参数配置方法只针对一个特定的Servlet有效,我们可以通过ServletContext来获取全局的、整个Web应用的初始化参数,全局的初始化参数是这样配置在web.xml文件中的:

<!-- 如果希望所有的Servlet都可以使用该配置,则必须这么做 --><context-param>    <param-name>name</param-name>    <param-value>gavin</param-value></context-param>

然后我们可以在任意一个Servlet中使用ServletContext获取这个参数:

String name = this.getServletContext().getInitParameter("name");


4、利用ServletContext对象读取资源文件(比如properties文件) 

读取资源文件要根据资源文件所在的位置来决定,一般分为以下两种情况:


4.1:文件在WebRoot文件夹下,即Web应用的根目录。这时候我们可以使用ServletContext来读取该资源文件。

假设我们Web根目录下有一个配置数据库信息的dbinfo.properties文件,里面配置了name和password属性,这时候可以通过ServletContext去读取这个文件:

// 这种方法的默认读取路径就是Web应用的根目录InputStream stream = this.getServletContext().getResourceAsStream("dbinfo.properties");// 创建属性对象Properties properties = new Properties();properties.load(stream);String name = properties.getProperty("name");String password = properties.getProperty("password");out.println("name="+name+";password="+password);

4.2:如果这个文件放在了src目录下,这时就不能用ServletContext来读取了,必须要使用类加载器去读取。

// 类加载器的默认读取路径是src根目录InputStream stream = MyServlet.class.getClassLoader().getResourceAsStream("dbinfo.properties")

如果这个文件此时还没有直接在src目录下,而是在src目录下的某个包下,比如在com.gavin包下,此时类加载器要加上包的路径,如下:

InputStream stream = MyServlet.class.getClassLoader().getResourceAsStream("com/gavin/dbinfo.properties")

补充一点,ServletContext只有在读取的文件在web应用的根目录下时,才能获取文件的全路径。比如我们在WebRoot文件夹下有一个images文件夹,images文件夹下有一个Servlet.jpg图片,为了得到这个图片的全路径,如下:

// 如何读取到一个文件的全路径,这里会得到在Tomcat的全路径String path = this.getServletContext().getRealPath("/images/Servlet.jpg");

在网站开发中,有很多功能要使用ServletContext,比如 

1. 网站计数器 

2. 网站的在线用户显示 

3. 简单的聊天系统

总之,如果是涉及到不同用户共享数据,而这些数据量不大,同时又不希望写入数据库中,我们就可以考虑使用ServletContext实现。


ServletContext使用建议

因为存在ServletContext中的数据在服务器中会长时间,这样就会占用很多内存,因此在使用ServletContext时,建议不要往里面添加过大的数据!


相关阅读:

Servlet教程

Servlet API官方文档


Servlet 数据库访问

本教程假定您已经了解了 JDBC 应用程序的工作方式。在您开始学习 Servlet 数据库访问之前,请确保您已经有适当的 JDBC 环境设置和数据库。

从基本概念下手,让我们来创建一个简单的表,并在表中创建几条记录。

创建数据库表

在测试数据库 TEST 中创建 Employees 表,请按以下步骤进行:

步骤 1:

打开命令行提示符(Command Prompt),并更改进入到安装目录,如下所示:

C:>C:>cd Program FilesMySQLinC:Program FilesMySQLin>

步骤 2:

登录到数据库,如下所示:

C:Program FilesMySQLin>mysql -u root -pEnter password: ********mysql>

步骤 3:

在测试数据库 TEST 中创建 Employee 表,如下所示:

mysql> use TEST;mysql> create table Employees    (     id int not null,     age int not null,     first varchar (255),     last varchar (255)    );Query OK, 0 rows affected (0.08 sec)mysql>

创建数据记录

最后,在 Employee 表中创建几条记录,如下所示:

mysql> INSERT INTO Employees VALUES (100, 18, 'Zara', 'Ali');Query OK, 1 row affected (0.05 sec) mysql> INSERT INTO Employees VALUES (101, 25, 'Mahnaz', 'Fatma');Query OK, 1 row affected (0.00 sec) mysql> INSERT INTO Employees VALUES (102, 30, 'Zaid', 'Khan');Query OK, 1 row affected (0.00 sec) mysql> INSERT INTO Employees VALUES (103, 28, 'Sumit', 'Mittal');Query OK, 1 row affected (0.00 sec) mysql>

访问数据库

下面的实例演示了如何使用 Servlet 访问 TEST 数据库。

// 加载必需的库import java.io.*;import java.util.*;import javax.servlet.*;import javax.servlet.http.*;import java.sql.*; public class DatabaseAccess extends HttpServlet{      public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // JDBC 驱动器名称和数据库的 URL      static final String JDBC_DRIVER="com.mysql.jdbc.Driver";        static final String DB_URL="jdbc:mysql://localhost/TEST";      //  数据库的用户名与密码,需要根据自己的设置      static final String USER = "root";      static final String PASS = "password";      // 设置响应内容类型      response.setContentType("text/html");      PrintWriter out = response.getWriter();      String title = "数据库结果";      String docType =        "<!doctype html public "-//w3c//dtd html 4.0 " +          "transitional//en">
";         out.println(docType +         "<html>
" +         "<head><title>" + title + "</title></head>
" +         "<body bgcolor="#f0f0f0">
" +         "<h1 align="center">" + title + "</h1>
");      try{         // 注册 JDBC 驱动器         Class.forName("com.mysql.jdbc.Driver");         // 打开一个连接         Connection conn = DriverManager.getConnection(DB_URL,USER,PASS);         // 执行 SQL 查询         Statement stmt = conn.createStatement();         String sql;         sql = "SELECT id, first, last, age FROM Employees";         ResultSet rs = stmt.executeQuery(sql);         // 展开结果集数据库         while(rs.next()){            // 通过字段检索            int id  = rs.getInt("id");            int age = rs.getInt("age");            String first = rs.getString("first");            String last = rs.getString("last");            // 输出数据            out.println("ID: " + id + "<br>");            out.println(", Age: " + age + "<br>");            out.println(", First: " + first + "<br>");            out.println(", Last: " + last + "<br>");         }         out.println("</body></html>");         // 完成后关闭         rs.close();         stmt.close();         conn.close();      }catch(SQLException se){         // 处理 JDBC 错误         se.printStackTrace();      }catch(Exception e){         // 处理 Class.forName 错误         e.printStackTrace();      }finally{         // 最后是用于关闭资源的块         try{            if(stmt!=null)               stmt.close();         }catch(SQLException se2){         }         try{            if(conn!=null)            conn.close();         }catch(SQLException se){            se.printStackTrace();         }      }   }} 

现在让我们来编译上面的 Servlet,并在 web.xml 文件中创建以下条目:

.... <servlet>     <servlet-name>DatabaseAccess</servlet-name>     <servlet-class>DatabaseAccess</servlet-class> </servlet>  <servlet-mapping>     <servlet-name>DatabaseAccess</servlet-name>     <url-pattern>/DatabaseAccess</url-pattern> </servlet-mapping>....

现在调用这个 Servlet,输入链接:http://localhost:8080/DatabaseAccess,将显示以下响应结果:

数据库结果ID: 100, Age: 18, First: Zara, Last: AliID: 101, Age: 25, First: Mahnaz, Last: FatmaID: 102, Age: 30, First: Zaid, Last: KhanID: 103, Age: 28, First: Sumit, Last: Mittal

Servlet 处理日期

使用 Servlet 的最重要的优势之一是,可以使用核心 Java 中的大多数可用的方法。本章将讲解 Java 提供的 java.util 包中的 Date 类,这个类封装了当前的日期和时间。

Date 类支持两个构造函数。第一个构造函数初始化当前日期和时间的对象。

Date( )

下面的构造函数接受一个参数,该参数等于 1970 年 1 月 1 日午夜以来经过的毫秒数。

Date(long millisec)

一旦您有一个可用的 Date 对象,您可以调用下列任意支持的方法来使用日期:

序号方法 & 描述
1boolean after(Date date)
如果调用的 Date 对象中包含的日期在 date 指定的日期之后,则返回 true,否则返回 false。
2boolean before(Date date)
如果调用的 Date 对象中包含的日期在 date 指定的日期之前,则返回 true,否则返回 false。
3Object clone( )
重复调用 Date 对象。
4int compareTo(Date date)
把调用对象的值与 date 的值进行比较。如果两个值是相等的,则返回 0。如果调用对象在 date 之前,则返回一个负值。如果调用对象在 date 之后,则返回一个正值。
5int compareTo(Object obj)
如果 obj 是 Date 类,则操作等同于 compareTo(Date)。否则,它会抛出一个 ClassCastException。
6boolean equals(Object date)
如果调用的 Date 对象中包含的时间和日期与 date 指定的相同,则返回 true,否则返回 false。
7long getTime( )
返回 1970 年 1 月 1 日以来经过的毫秒数。
8int hashCode( )
为调用对象返回哈希代码。
9void setTime(long time)
设置 time 指定的时间和日期,这表示从 1970 年 1 月 1 日午夜以来经过的时间(以毫秒为单位)。
10String toString( )
转换调用的 Date 对象为一个字符串,并返回结果。

获取当前的日期和时间

在 Java Servlet 中获取当前的日期和时间是非常容易的。您可以使用一个简单的 Date 对象的 toString() 方法来输出当前的日期和时间,如下所示:

// 导入必需的 java 库import java.io.*;import java.util.Date;import javax.servlet.*;import javax.servlet.http.*; // 扩展 HttpServlet 类public class CurrentDate extends HttpServlet {   public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 设置响应内容类型      response.setContentType("text/html");       PrintWriter out = response.getWriter();      String title = "显示当前的日期和时间";      Date date = new Date();      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +        "<html>
" +        "<head><title>" + title + "</title></head>
" +        "<body bgcolor="#f0f0f0">
" +        "<h1 align="center">" + title + "</h1>
" +        "<h2 align="center">" + date.toString() + "</h2>
" +        "</body></html>");  }}

现在,让我们来编译上面的 Servlet,并在 web.xml 文件中创建适当的条目,然后通过访问 http://localhost:8080/CurrentDate 来调用该 Servlet。这将会产生如下的结果:

显示当前的日期和时间

Mon Jun 21 21:46:49 GMT+04:00 2010

尝试刷新 URL http://localhost:8080/CurrentDate,每隔几秒刷新一次您都会发现显示时间的差异。

日期比较

正如上面所提到的,您可以在 Servlet 中使用所有可用的 Java 方法。如果您需要比较两个日期,以下是方法:

  • 您可以使用 getTime() 来获取两个对象自 1970 年 1 月 1 日午夜以来经过的时间(以毫秒为单位),然后对这两个值进行比较。
  • 您可以使用方法 before( )、after( ) 和 equals( )。由于一个月里 12 号在 18 号之前,例如,new Date(99, 2, 12).before(new Date (99, 2, 18)) 返回 true。
  • 您可以使用 compareTo( ) 方法,该方法由 Comparable 接口定义,由 Date 实现。

使用 SimpleDateFormat 格式化日期

SimpleDateFormat 是一个以语言环境敏感的方式来格式化和解析日期的具体类。 SimpleDateFormat 允许您选择任何用户定义的日期时间格式化的模式。

让我们修改上面的实例,如下所示:

// 导入必需的 java 库import java.io.*;import java.text.*;import java.util.Date;import javax.servlet.*;import javax.servlet.http.*; // 扩展 HttpServlet 类public class CurrentDate extends HttpServlet {   public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 设置响应内容类型      response.setContentType("text/html");       PrintWriter out = response.getWriter();      String title = "显示当前的日期和时间";      Date dNow = new Date( );      SimpleDateFormat ft =       new SimpleDateFormat ("E yyyy.MM.dd 'at' hh:mm:ss a zzz");      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +        "<html>
" +        "<head><title>" + title + "</title></head>
" +        "<body bgcolor="#f0f0f0">
" +        "<h1 align="center">" + title + "</h1>
" +        "<h2 align="center">" + ft.format(dNow) + "</h2>
" +        "</body></html>");  }}

再次编译上面的 Servlet,然后通过访问 http://localhost:8080/CurrentDate 来调用该 Servlet。这将会产生如下的结果:

显示当前的日期和时间

Mon 2010.06.21 at 10:06:44 PM GMT+04:00


简单的日期格式的格式代码

使用事件模式字符串来指定时间格式。在这种模式下,所有的 ASCII 字母被保留为模式字母,这些字母定义如下:

字符描述实例
GEra 指示器AD
y四位数表示的年2001
M一年中的月July 或 07
d一月中的第几天10
h带有 A.M./P.M. 的小时(1~12)12
H一天中的第几小时(0~23)22
m一小时中的第几分30
s一分中的第几秒55
S毫秒234
E一周中的星期几Tuesday
D一年中的第几天360
F所在的周是这个月的第几周2 (second Wed. in July)
w一年中的第几周40
W一月中的第几周1
aA.M./P.M. 标记PM
k一天中的第几小时(1~24)24
K带有 A.M./P.M. 的小时(0~11)10
z时区Eastern Standard Time
'Escape for textDelimiter
"单引号`

如需查看可用的处理日期方法的完整列表,您可以参考标准的 Java 文档。

Servlet 网页重定向

当文档移动到新的位置,我们需要向客户端发送这个新位置时,我们需要用到网页重定向。当然,也可能是为了负载均衡,或者只是为了简单的随机,这些情况都有可能用到网页重定向。

重定向请求到另一个网页的最简单的方式是使用 response 对象的 sendRedirect() 方法。下面是该方法的定义: 将请求重定向到另一页的最简单的方法是,用方法的sendRedirect()的响应对象。以下是这种方法的定义:

public void HttpServletResponse.sendRedirect(String location)throws IOException 

该方法把响应连同状态码和新的网页位置发送回浏览器。您也可以通过把 setStatus() 和 setHeader() 方法一起使用来达到同样的效果:

....String site = "http://www.newpage.com" ;response.setStatus(response.SC_MOVED_TEMPORARILY);response.setHeader("Location", site); ....

实例

本实例显示了 Servlet 如何进行页面重定向到另一个位置:

import java.io.*;import java.sql.Date;import java.util.*;import javax.servlet.*;import javax.servlet.http.*;public class PageRedirect extends HttpServlet{      public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 设置响应内容类型      response.setContentType("text/html");      // 要重定向的新位置      String site = new String("http://www.51coolma.cn");      response.setStatus(response.SC_MOVED_TEMPORARILY);      response.setHeader("Location", site);        }} 

现在让我们来编译上面的 Servlet,并在 web.xml 文件中创建以下条目:

.... <servlet>     <servlet-name>PageRedirect</servlet-name>     <servlet-class>PageRedirect</servlet-class> </servlet> <servlet-mapping>     <servlet-name>PageRedirect</servlet-name>     <url-pattern>/PageRedirect</url-pattern> </servlet-mapping>....

现在通过访问 URL http://localhost:8080/PageRedirect 来调用这个 Servlet。这将把您转到给定的 URL http://www.51coolma.cn。

Servlet 点击计数器

网页点击计数器

很多时候,您可能有兴趣知道网站的某个特定页面上的总点击量。使用 Servlet 来计算这些点击量是非常简单的,因为一个 Servlet 的生命周期是由它运行所在的容器控制的。

以下是实现一个简单的基于 Servlet 生命周期的网页点击计数器需要采取的步骤:

  • 在 init() 方法中初始化一个全局变量。
  • 每次调用 doGet() 或 doPost() 方法时,都增加全局变量。
  • 如果需要,您可以使用一个数据库表来存储全局变量的值在 destroy() 中。在下次初始化 Servlet 时,该值可在 init() 方法内被读取。这一步是可选的。
  • 如果您只想对一个 session 会话计数一次页面点击,那么请使用 isNew() 方法来检查该 session 会话是否已点击过相同页面。这一步是可选的。
  • 您可以通过显示全局计数器的值,来在网站上展示页面的总点击量。这一步是可选的。

在这里,我们假设 Web 容器将无法重新启动。如果是重新启动或 Servlet 被销毁,计数器将被重置。

实例

本实例演示了如何实现一个简单的网页点击计数器:

import java.io.*;import java.sql.Date;import java.util.*;import javax.servlet.*;import javax.servlet.http.*;public class PageHitCounter extends HttpServlet{      private int hitCount;                  public void init()   {      // 重置点击计数器     hitCount = 0;  }   public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 设置响应内容类型      response.setContentType("text/html");      // 该方法在 Servlet 被点击时执行       // 增加 hitCount       hitCount++;       PrintWriter out = response.getWriter();      String title = "总点击量";      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +        "<html>
" +        "<head><title>" + title + "</title></head>
" +        "<body bgcolor="#f0f0f0">
" +        "<h1 align="center">" + title + "</h1>
" +        "<h2 align="center">" + hitCount + "</h2>
" +        "</body></html>");  }  public void destroy()   {       // 这一步是可选的,但是如果需要,您可以把 hitCount 的值写入到数据库  } } 

现在让我们来编译上面的 Servlet,并在 web.xml 文件中创建以下条目:

.... <servlet>     <servlet-name>PageHitCounter</servlet-name>     <servlet-class>PageHitCounter</servlet-class> </servlet> <servlet-mapping>     <servlet-name>PageHitCounter</servlet-name>     <url-pattern>/PageHitCounter</url-pattern> </servlet-mapping>....

现在通过访问 URL http://localhost:8080/PageHitCounter 来调用这个 Servlet。这将会在每次页面刷新时,把计数器的值增加 1,结果如下所示:

总点击量

6


网站点击计数器

很多时候,您可能有兴趣知道整个网站的总点击量。在 Servlet 中,这也是非常简单的,我们可以使用过滤器做到这一点。

以下是实现一个简单的基于过滤器生命周期的网站点击计数器需要采取的步骤:

  • 在过滤器的 init() 方法中初始化一个全局变量。
  • 每次调用 doFilter 方法时,都增加全局变量。
  • 如果需要,您可以使用一个数据库表来存储全局变量的值在过滤器的 destroy() 中。在下次初始化过滤器时,该值可在 init() 方法内被读取。这一步是可选的。

在这里,我们假设 Web 容器将无法重新启动。如果是重新启动或 Servlet 被销毁,点击计数器将被重置。

实例

本实例演示了如何实现一个简单的网站点击计数器:

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*;import java.util.*;public class SiteHitCounter implements Filter{      private int hitCount;                  public void  init(FilterConfig config)                     throws ServletException{     // 重置点击计数器     hitCount = 0;  }  public void  doFilter(ServletRequest request,               ServletResponse response,              FilterChain chain)               throws java.io.IOException, ServletException {      // 把计数器的值增加 1      hitCount++;      // 输出计数器      System.out.println("网站访问统计:"+ hitCount );      // 把请求传回到过滤器链      chain.doFilter(request,response);  }  public void destroy()   {       // 这一步是可选的,但是如果需要,您可以把 hitCount 的值写入到数据库  } } 

现在让我们来编译上面的 Servlet,并在 web.xml 文件中创建以下条目:

....<filter>   <filter-name>SiteHitCounter</filter-name>   <filter-class>SiteHitCounter</filter-class></filter><filter-mapping>   <filter-name>SiteHitCounter</filter-name>   <url-pattern>/*</url-pattern></filter-mapping>....

现在访问网站的任意页面,比如 http://localhost:8080/。这将会在每次任意页面被点击时,把计数器的值增加 1,它会在日志中显示以下消息:

网站访问统计: 1网站访问统计: 2网站访问统计: 3网站访问统计: 4网站访问统计: 5..................

Servlet 自动刷新页面

假设有一个网页,它是显示现场比赛成绩或股票市场状况或货币兑换率。对于所有这些类型的页面,您需要定期刷新网页。

Java Servlet 提供了一个机制,使得网页会在给定的时间间隔自动刷新。

刷新网页的最简单的方式是使用响应对象的方法 setIntHeader()。以下是这种方法的定义:

public void setIntHeader(String header, int headerValue)

此方法把头信息 "Refresh" 连同一个表示时间间隔的整数值(以秒为单位)发送回浏览器。

自动刷新页面实例

本实例演示了 Servlet 如何使用 setIntHeader() 方法来设置 Refresh 头信息,从而实现自动刷新页面。

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*;import java.util.*; // 扩展 HttpServlet 类public class Refresh extends HttpServlet {   // 处理 GET 方法请求的方法  public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 设置刷新自动加载的事件间隔为 5 秒      response.setIntHeader("Refresh", 5);       // 设置响应内容类型      response.setContentType("text/html");       // 获取当前的时间      Calendar calendar = new GregorianCalendar();      String am_pm;      int hour = calendar.get(Calendar.HOUR);      int minute = calendar.get(Calendar.MINUTE);      int second = calendar.get(Calendar.SECOND);      if(calendar.get(Calendar.AM_PM) == 0)        am_pm = "AM";      else        am_pm = "PM";       String CT = hour+":"+ minute +":"+ second +" "+ am_pm;          PrintWriter out = response.getWriter();      String title = "使用 Servlet 自动刷新页面";      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +        "<html>
" +        "<head><title>" + title + "</title></head>
"+        "<body bgcolor="#f0f0f0">
" +        "<h1 align="center">" + title + "</h1>
" +        "<p>当前时间是:" + CT + "</p>
");  }  // 处理 POST 方法请求的方法  public void doPost(HttpServletRequest request,                     HttpServletResponse response)      throws ServletException, IOException {     doGet(request, response);  }}

现在让我们来编译上面的 Servlet,并在 web.xml 文件中创建以下条目:

.... <servlet>     <servlet-name>Refresh</servlet-name>     <servlet-class>Refresh</servlet-class> </servlet>  <servlet-mapping>     <servlet-name>Refresh</servlet-name>     <url-pattern>/Refresh</url-pattern> </servlet-mapping>....

现在通过访问 URL http://localhost:8080/Refresh 来调用这个 Servlet。这将会每隔 5 秒钟显示一次当前系统时间。运行该 Servlet,并等待查看结果:

使用 Servlet 自动刷新页面

当前时间是:9:44:50 PM

Servlet 发送电子邮件

使用 Servlet 发送一封电子邮件是很简单的,但首先您必须在您的计算机上安装 JavaMail APIJava Activation Framework)JAF)

下载并解压缩这些文件,在新创建的顶层目录中,您会发现这两个应用程序的一些 jar 文件。您需要把 mail.jaractivation.jar 文件添加到您的 CLASSPATH 中。

发送一封简单的电子邮件

下面的实例将从您的计算机上发送一封简单的电子邮件。这里假设您的本地主机已连接到互联网,并支持发送电子邮件。同时确保 Java Email API 包和 JAF 包的所有的 jar 文件在 CLASSPATH 中都是可用的。

// 文件名 SendEmail.javaimport java.io.*;import java.util.*;import javax.servlet.*;import javax.servlet.http.*;import javax.mail.*;import javax.mail.internet.*;import javax.activation.*; public class SendEmail extends HttpServlet{      public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 收件人的电子邮件 ID      String to = "abcd@gmail.com";       // 发件人的电子邮件 ID      String from = "web@gmail.com";       // 假设您是从本地主机发送电子邮件      String host = "localhost";       // 获取系统的属性      Properties properties = System.getProperties();       // 设置邮件服务器      properties.setProperty("mail.smtp.host", host);       // 获取默认的 Session 对象      Session session = Session.getDefaultInstance(properties);         // 设置响应内容类型      response.setContentType("text/html");      PrintWriter out = response.getWriter();      try{         // 创建一个默认的 MimeMessage 对象         MimeMessage message = new MimeMessage(session);         // 设置 From: header field of the header.         message.setFrom(new InternetAddress(from));         // 设置 To: header field of the header.         message.addRecipient(Message.RecipientType.TO,                                  new InternetAddress(to));         // 设置 Subject: header field         message.setSubject("This is the Subject Line!");         // 现在设置实际消息         message.setText("This is actual message");         // 发送消息         Transport.send(message);         String title = "发送电子邮件";         String res = "成功发送消息...";         String docType =         "<!doctype html public "-//w3c//dtd html 4.0 " +          "transitional//en">
";         out.println(docType +         "<html>
" +         "<head><title>" + title + "</title></head>
" +         "<body bgcolor="#f0f0f0">
" +         "<h1 align="center">" + title + "</h1>
" +         "<p align="center">" + res + "</p>
" +         "</body></html>");      }catch (MessagingException mex) {         mex.printStackTrace();      }   }} 

现在让我们来编译上面的 Servlet,并在 web.xml 文件中创建以下条目:

.... <servlet>     <servlet-name>SendEmail</servlet-name>     <servlet-class>SendEmail</servlet-class> </servlet>  <servlet-mapping>     <servlet-name>SendEmail</servlet-name>     <url-pattern>/SendEmail</url-pattern> </servlet-mapping>....

现在通过访问 URL http://localhost:8080/SendEmail 来调用这个 Servlet。这将会发送一封电子邮件到给定的电子邮件 ID abcd@gmail.com,并将显示下面所示的响应:

发送电子邮件

成功发送消息...

如果您想要发送一封电子邮件给多个收件人,那么请使用下面的方法来指定多个电子邮件 ID:

void addRecipients(Message.RecipientType type,                    Address[] addresses)throws MessagingException

下面是对参数的描述:

  • type:这将被设置为 TO、CC 或 BCC。在这里,CC 代表抄送,BCC 代表密件抄送。例如 Message.RecipientType.TO
  • addresses:这是电子邮件 ID 的数组。当指定电子邮件 ID 时,您需要使用 InternetAddress() 方法。

发送一封 HTML 电子邮件

下面的实例将从您的计算机上发送一封 HTML 格式的电子邮件。这里假设您的本地主机已连接到互联网,并支持发送电子邮件。同时确保 Java Email API 包和 JAF 包的所有的 jar 文件在 CLASSPATH 中都是可用的。

本实例与上一个实例很类似,但是这里我们使用 setContent() 方法来设置第二个参数为 "text/html" 的内容,该参数用来指定 HTML 内容是包含在消息中的。

使用这个实例,您可以发送内容大小不限的 HTML 内容。

// 文件名 SendEmail.javaimport java.io.*;import java.util.*;import javax.servlet.*;import javax.servlet.http.*;import javax.mail.*;import javax.mail.internet.*;import javax.activation.*; public class SendEmail extends HttpServlet{      public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 收件人的电子邮件 ID      String to = "abcd@gmail.com";       // 发件人的电子邮件 ID      String from = "web@gmail.com";       // 假设您是从本地主机发送电子邮件      String host = "localhost";       // 获取系统的属性      Properties properties = System.getProperties();       // 设置邮件服务器      properties.setProperty("mail.smtp.host", host);       // 获取默认的 Session 对象      Session session = Session.getDefaultInstance(properties);            // 设置响应内容类型      response.setContentType("text/html");      PrintWriter out = response.getWriter();      try{         // 创建一个默认的 MimeMessage 对象         MimeMessage message = new MimeMessage(session);         // 设置 From: header field of the header.         message.setFrom(new InternetAddress(from));         // 设置 To: header field of the header.         message.addRecipient(Message.RecipientType.TO,                                  new InternetAddress(to));         // 设置 Subject: header field         message.setSubject("This is the Subject Line!");         // 设置实际的 HTML 消息,内容大小不限         message.setContent("<h1>This is actual message</h1>",                            "text/html" );         // 发送消息         Transport.send(message);         String title = "发送电子邮件";         String res = "成功发送消息...";         String docType =         "<!doctype html public "-//w3c//dtd html 4.0 " +          "transitional//en">
";         out.println(docType +         "<html>
" +         "<head><title>" + title + "</title></head>
" +         "<body bgcolor="#f0f0f0">
" +         "<h1 align="center">" + title + "</h1>
" +         "<p align="center">" + res + "</p>
" +         "</body></html>");      }catch (MessagingException mex) {         mex.printStackTrace();      }   }} 

编译并运行上面的 Servlet ,在给定的电子邮件 ID 上发送 HTML 消息。

在电子邮件中发送附件

下面的实例将从您的计算机上发送一封带有附件的电子邮件。这里假设您的本地主机已连接到互联网,并支持发送电子邮件。同时确保 Java Email API 包和 JAF 包的所有的 jar 文件在 CLASSPATH 中都是可用的。

// 文件名 SendEmail.javaimport java.io.*;import java.util.*;import javax.servlet.*;import javax.servlet.http.*;import javax.mail.*;import javax.mail.internet.*;import javax.activation.*; public class SendEmail extends HttpServlet{      public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 收件人的电子邮件 ID      String to = "abcd@gmail.com";       // 发件人的电子邮件 ID      String from = "web@gmail.com";       // 假设您是从本地主机发送电子邮件      String host = "localhost";       // 获取系统的属性      Properties properties = System.getProperties();       // 设置邮件服务器      properties.setProperty("mail.smtp.host", host);       // 获取默认的 Session 对象      Session session = Session.getDefaultInstance(properties);           // 设置响应内容类型      response.setContentType("text/html");      PrintWriter out = response.getWriter();       try{         // 创建一个默认的 MimeMessage 对象         MimeMessage message = new MimeMessage(session);          // 设置 From: header field of the header.         message.setFrom(new InternetAddress(from));          // 设置 To: header field of the header.         message.addRecipient(Message.RecipientType.TO,                                  new InternetAddress(to));          // 设置 Subject: header field         message.setSubject("This is the Subject Line!");          // 创建消息部分          BodyPart messageBodyPart = new MimeBodyPart();          // 填写消息         messageBodyPart.setText("This is message body");                  // 创建一个多部分消息         Multipart multipart = new MimeMultipart();          // 设置文本消息部分         multipart.addBodyPart(messageBodyPart);          // 第二部分是附件         messageBodyPart = new MimeBodyPart();         String filename = "file.txt";         DataSource source = new FileDataSource(filename);         messageBodyPart.setDataHandler(new DataHandler(source));         messageBodyPart.setFileName(filename);         multipart.addBodyPart(messageBodyPart);          // 发送完整的消息部分         message.setContent(multipart );          // 发送消息         Transport.send(message);         String title = "发送电子邮件";         String res = "成功发送电子邮件...";         String docType =         "<!doctype html public "-//w3c//dtd html 4.0 " +          "transitional//en">
";         out.println(docType +         "<html>
" +         "<head><title>" + title + "</title></head>
" +         "<body bgcolor="#f0f0f0">
" +         "<h1 align="center">" + title + "</h1>
" +         "<p align="center">" + res + "</p>
" +         "</body></html>");      }catch (MessagingException mex) {         mex.printStackTrace();      }   }} 

编译并运行上面的 Servlet ,在给定的电子邮件 ID 上发送带有文件附件的消息。

用户身份认证部分

如果需要向电子邮件服务器提供用户 ID 和密码进行身份认证,那么您可以设置如下属性:

 props.setProperty("mail.user", "myuser"); props.setProperty("mail.password", "mypwd");

电子邮件发送机制的其余部分与上面讲解的保持一致。

Servlet

涉及到 WEB-INF 子目录的 Web 应用程序结构是所有的 Java web 应用程序的标准,并由 Servlet API 规范指定。给定一个顶级目录名 myapp,目录结构如下所示:

/myapp    /images    /WEB-INF        /classes        /lib

WEB-INF 子目录中包含应用程序的部署描述符,名为 web.xml。所有的 HTML 文件都位于顶级目录 myapp 下。对于 admin 用户,您会发现 ROOT 目录是 myApp 的父目录。

创建包中的 Servlet

WEB-INF/classes 目录包含了所有的 Servlet 类和其他类文件,类文件所在的目录结构与他们的包名称匹配。例如,如果您有一个完全合格的类名称 com.myorg.MyServlet,那么这个 Servlet 类必须位于以下目录中:

/myapp/WEB-INF/classes/com/myorg/MyServlet.class

下面的例子创建包名为 com.myorg 的 MyServlet 类。

// 为包命名package com.myorg;  // 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*; public class MyServlet extends HttpServlet {   private String message;   public void init() throws ServletException  {      // 执行必需的的初始化      message = "Hello World";  }   public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 设置响应内容类型      response.setContentType("text/html");       // 实际的逻辑是在这里      PrintWriter out = response.getWriter();      out.println("<h1>" + message + "</h1>");  }    public void destroy()  {      // 什么也不做  }}

编译包中的 Servlet

编译包中的类与编译其他的类没有什么大的不同。最简单的方法是让您的 java 文件保留完全限定路径,如上面提到的类,将被保留在 com.myorg 中。您还需要在 CLASSPATH 中添加该目录。

假设您的环境已正确设置,进入 <Tomcat-installation-directory>/webapps/ROOT/WEB-INF/classes 目录,并编译 MyServlet.java,如下所示:

$ javac MyServlet.java

如果 Servlet 依赖于其他库,那么您必须在 CLASSPATH 中也要引用那些 JAR 文件。这里我只引用了 servlet-api.jar JAR 文件,因为我在 Hello World 程序中并没有使用任何其他库。

该命令行使用内置的 javac 编译器,它是 Sun Microsystems Java 软件开发工具包(JDK,全称 Java Software Development Kit)附带的。 Microsystems的Java软件开发工具包(JDK)。为了让该命令正常工作,必须包括您在 PATH 环境变量中所使用的 Java SDK 的位置。

如果一切顺利,上述编译会在同一目录下生成 MyServlet.class 文件。下一节将解释如何把一个已编译的 Servlet 部署到生产中。

Servlet 打包部署

默认情况下,Servlet 应用程序位于路径 <Tomcat-installation-directory>/webapps/ROOT 下,且类文件放在 <Tomcat-installation-directory>/webapps/ROOT/WEB-INF/classes 中。

如果您有一个完全合格的类名称 com.myorg.MyServlet,那么这个 Servlet 类必须位于 WEB-INF/classes/com/myorg/MyServlet.class 中,您需要在位于 <Tomcat-installation-directory>/webapps/ROOT/WEB-INF/ 的 web.xml 文件中创建以下条目:

    <servlet>        <servlet-name>MyServlet</servlet-name>        <servlet-class>com.myorg.MyServlet</servlet-class>    </servlet>     <servlet-mapping>        <servlet-name>MyServlet</servlet-name>        <url-pattern>/MyServlet</url-pattern>    </servlet-mapping>

上面的条目要被创建在 web.xml 文件中的 <web-app>...</web-app> 标签内。在该文件中可能已经有各种可用的条目,但不要在意。

到这里,您基本上已经完成了,现在让我们使用 <Tomcat-installation-directory>instartup.bat(在 Windows 上)或 <Tomcat-installation-directory>/bin/startup.sh(在 Linux/Solaris 等上)启动 tomcat 服务器,最后在浏览器的地址栏中输入 http://localhost:8080/MyServlet。如果一切顺利,您会看到下面的结果:

Hello World

Servlet 调试

测试/调试 Servlet 始终是开发使用过程中的难点。Servlet 往往涉及大量的客户端/服务器交互,可能会出现错误但又难以重现。

这里有一些提示和建议,可以帮助您调试。

System.out.println()

System.out.println() 是作为一个标记来使用的,用来测试一段特定的代码是否被执行。我们也可以打印出变量的值。此外:

  • 由于 System 对象是核心 Java 对象的一部分,它可以在不需要安装任何额外类的情况下被用于任何地方。这包括 Servlet、JSP、RMI、EJB's、普通的 Beans 和类,以及独立的应用程序。
  • 与在断点处停止不同,写入到 System.out 不会干扰到应用程序的正常执行流程,这使得它在时序是至关重要的时候显得尤为有价值。

下面是使用 System.out.println() 的语法:

System.out.println("Debugging message");

通过上面的语法生成的所有消息将被记录在 Web 服务器日志文件中。

消息日志

使用适当的日志记录方法来记录所有调试、警告和错误消息,这是非常好的想法,推荐使用 log4J 来记录所有的消息。

Servlet API 还提供了一个简单的输出信息的方式,使用 log() 方法,如下所示:

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*;public class ContextLog extends HttpServlet {  public void doGet(HttpServletRequest request,       HttpServletResponse response) throws ServletException,         java.io.IOException {          String par = request.getParameter("par1");      // 调用两个 ServletContext.log 方法      ServletContext context = getServletContext( );      if (par == null || par.equals(""))      // 通过 Throwable 参数记录版本      context.log("No message received:",          new IllegalStateException("Missing parameter"));      else          context.log("Here is the visitor's message: " + par);            response.setContentType("text/html");      java.io.PrintWriter out = response.getWriter( );      String title = "Context Log";      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +        "<html>
" +        "<head><title>" + title + "</title></head>
" +        "<body bgcolor="#f0f0f0">
" +        "<h1 align="center">" + title + "</h1>
" +        "<h2 align="center">Messages sent</h2>
" +        "</body></html>");    } //doGet}

ServletContext 把它的文本消息记录到 Servlet 容器的日志文件中。对于 Tomcat,这些日志可以在 <Tomcat-installation-directory>/logs 目录中找到。

这些日志文件确实对新出现的错误或问题的频率给出指示。正因为如此,建议在通常不会发生的异常的 catch 子句中使用 log() 函数。

使用 JDB 调试器

您可以使用调试 applet 或应用程序的 jdb 命令来调试 Servlet。

为了调试一个 Servlet,我们可以调试 sun.servlet.http.HttpServer,然后把它看成是 HttpServer 执行 Servlet 来响应浏览器端的 HTTP 请求。这与调试 applet 小程序非常相似。与调试 applet 不同的是,实际被调试的程序是 sun.applet.AppletViewer。

大多数调试器会自动隐藏如何调试 applet 的细节。同样的,对于 servlet,您必须帮调试器执行以下操作:

  • 设置您的调试器的类路径 classpath,以便它可以找到 sun.servlet.http.Http-Server 和相关的类。
  • 设置您的调试器的类路径 classpath,以便它可以找到您的 servlet 和支持的类,通常是在 server_root/servlets 和 server_root/classes 中。

您通常不会希望 server_root/servlets 在您的 classpath 中,因为它会禁用 servlet 的重新加载。但是这种包含规则对于调试是非常有用的。它允许您的调试器在 HttpServer 中的自定义 Servlet 加载器加载 Servlet 之前在 Servlet 中设置断点。

如果您已经设置了正确的类路径 classpath,就可以开始调试 sun.servlet.http.HttpServer。可以在您想要调试的 Servlet 代码中设置断点,然后通过 Web 浏览器使用给定的 Servlet(http://localhost:8080/servlet/ServletToDebug)向 HttpServer 发出请求。您会看到程序执行到断点处会停止。

使用注释

代码中的注释有助于以各种方式进行调试。注释可用于调试过程的很多其他方式中。

该 Servlet 使用 Java 注释和单行注释(//...),多行注释(/* ...*/)可用于暂时移除部分 Java 代码。如果 bug 消失,仔细看看您刚才注释的代码并找出问题所在。

客户端和服务器端头信息

有时,当一个 Servlet 并没有像预期那样时,查看原始的 HTTP 请求和响应是非常有用的。如果您熟悉 HTTP 结构,您可以阅读请求和响应,看看这些头信息究竟是什么。

重要的调试技巧

下面列出了一些 Servlet 调试的技巧:

  • 请注意,server_root/classes 不会重载,而 server_root/servlets 可能会。
  • 要求浏览器显示它所显示的页面的原始内容。这有助于识别格式的问题。它通常是"视图"菜单下的一个选项。
  • 通过强制执行完全重新加载页面来确保浏览器还没有缓存前一个请求的输出。在 Netscape Navigator 中,请使用 Shift-Reload,在 Internet Explorer 中,请使用 Shift-Refresh。
  • 请确认 servlet 的 init() 方法接受一个 ServletConfig 参数,并调用 super.init(config)。

Servlet 国际化

在我们开始之前,先来看看三个重要术语:

  • 国际化(i18n):这意味着一个网站提供了不同版本的翻译成访问者的语言或国籍的内容。
  • 本地化(l10n):这意味着向网站添加资源,以使其适应特定的地理或文化区域,例如网站翻译成印地文(Hindi)。
  • 区域设置(locale):这是一个特殊的文化或地理区域。它通常指语言符号后跟一个下划线和一个国家符号。例如 "en_US" 表示针对 US 的英语区域设置。

当建立一个全球性的网站时有一些注意事项。本教程不会讲解这些注意事项的完整细节,但它会通过一个很好的实例向您演示如何通过差异化定位(即区域设置)来让网页以不同语言呈现。

Servlet 可以根据请求者的区域设置拾取相应版本的网站,并根据当地的语言、文化和需求提供相应的网站版本。以下是 request 对象中返回 Locale 对象的方法。

java.util.Locale request.getLocale() 

检测区域设置

下面列出了重要的区域设置方法,您可以使用它们来检测请求者的地理位置、语言和区域设置。下面所有的方法都显示了请求者浏览器中设置的国家名称和语言名称。

序号方法 & 描述
1String getCountry()
该方法以 2 个大写字母形式的 ISO 3166 格式返回该区域设置的国家/地区代码。
2String getDisplayCountry()
该方法返回适合向用户显示的区域设置的国家的名称。
3String getLanguage()
该方法以小写字母形式的 ISO 639 格式返回该区域设置的语言代码。
4String getDisplayLanguage()
该方法返回适合向用户显示的区域设置的语言的名称。
5String getISO3Country()
该方法返回该区域设置的国家的三个字母缩写。
6String getISO3Language()
该方法返回该区域设置的语言的三个字母的缩写。

实例

本实例演示了如何显示某个请求的语言和相关的国家:

import java.io.*;import javax.servlet.*;import javax.servlet.http.*;import java.util.Locale;public class GetLocale extends HttpServlet{      public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 获取客户端的区域设置      Locale locale = request.getLocale();      String language = locale.getLanguage();      String country = locale.getCountry();      // 设置响应内容类型      response.setContentType("text/html");      PrintWriter out = response.getWriter();      String title = "检测区域设置";      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +        "<html>
" +        "<head><title>" + title + "</title></head>
" +        "<body bgcolor="#f0f0f0">
" +        "<h1 align="center">" + language + "</h1>
" +        "<h2 align="center">" + country + "</h2>
" +        "</body></html>");  }} 

语言设置

Servlet 可以输出以西欧语言(如英语、西班牙语、德语、法语、意大利语、荷兰语等)编写的页面。在这里,为了能正确显示所有的字符,设置 Content-Language 头是非常重要的。

第二点是使用 HTML 实体显示所有的特殊字符,例如,"ñ" 表示 "ñ","¡" 表示 "¡",如下所示:

import java.io.*;import javax.servlet.*;import javax.servlet.http.*;import java.util.Locale;public class DisplaySpanish extends HttpServlet{      public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {    // 设置响应内容类型    response.setContentType("text/html");    PrintWriter out = response.getWriter();    // 设置西班牙语言代码    response.setHeader("Content-Language", "es");    String title = "En Espa&ntilde;ol";    String docType =     "<!doctype html public "-//w3c//dtd html 4.0 " +      "transitional//en">
";     out.println(docType +     "<html>
" +     "<head><title>" + title + "</title></head>
" +     "<body bgcolor="#f0f0f0">
" +     "<h1>" + "En Espa&ntilde;ol:" + "</h1>
" +     "<h1>" + "&iexcl;Hola Mundo!" + "</h1>
" +     "</body></html>");  }} 

特定于区域设置的日期

您可以使用 java.text.DateFormat 类及其静态方法 getDateTimeInstance() 来格式化特定于区域设置的日期和时间。下面的实例演示了如何格式化特定于某个给定的区域设置的日期:

import java.io.*;import javax.servlet.*;import javax.servlet.http.*;import java.util.Locale;import java.text.DateFormat;import java.util.Date;public class DateLocale extends HttpServlet{      public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {    // 设置响应内容类型    response.setContentType("text/html");    PrintWriter out = response.getWriter();    // 获取客户端的区域设置    Locale locale = request.getLocale( );    String date = DateFormat.getDateTimeInstance(                                  DateFormat.FULL,                                   DateFormat.SHORT,                                   locale).format(new Date( ));    String title = "特定于区域设置的日期";    String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +      "<html>
" +      "<head><title>" + title + "</title></head>
" +      "<body bgcolor="#f0f0f0">
" +      "<h1 align="center">" + date + "</h1>
" +      "</body></html>");  }} 

特定于区域设置的货币

您可以使用 java.text.NumberFormat 类及其静态方法 getCurrencyInstance() 来格式化数字(比如 long 类型或 double 类型)为特定于区域设置的货币。下面的实例演示了如何格式化特定于某个给定的区域设置的货币:

import java.io.*;import javax.servlet.*;import javax.servlet.http.*;import java.util.Locale;import java.text.NumberFormat;import java.util.Date;public class CurrencyLocale extends HttpServlet{      public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {    // 设置响应内容类型    response.setContentType("text/html");    PrintWriter out = response.getWriter();    // 获取客户端的区域设置    Locale locale = request.getLocale( );    NumberFormat nft = NumberFormat.getCurrencyInstance(locale);    String formattedCurr = nft.format(1000000);    String title = "特定于区域设置的货币";    String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +      "<html>
" +      "<head><title>" + title + "</title></head>
" +      "<body bgcolor="#f0f0f0">
" +      "<h1 align="center">" + formattedCurr + "</h1>
" +      "</body></html>");  }} 

特定于区域设置的百分比

您可以使用 java.text.NumberFormat 类及其静态方法 getPercentInstance() 来格式化特定于区域设置的百分比。下面的实例演示了如何格式化特定于某个给定的区域设置的百分比:

import java.io.*;import javax.servlet.*;import javax.servlet.http.*;import java.util.Locale;import java.text.NumberFormat;import java.util.Date;public class PercentageLocale extends HttpServlet{      public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {    // 设置响应内容类型    response.setContentType("text/html");    PrintWriter out = response.getWriter();    // 获取客户端的区域设置    Locale locale = request.getLocale( );    NumberFormat nft = NumberFormat.getPercentInstance(locale);    String formattedPerc = nft.format(0.51);    String title = "特定于区域设置的百分比";    String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +      "<html>
" +      "<head><title>" + title + "</title></head>
" +      "<body bgcolor="#f0f0f0">
" +      "<h1 align="center">" + formattedPerc + "</h1>
" +      "</body></html>");  }} 

Servlet容器主要是JavaWeb应用提供运行时环境,所以也可以称之为JavaWeb应用容器,或者Servlet/JSP容器。Servlet容器主要负责管理Servlet、JSP的生命周期以及它们的共享数据。


Servlet容器有哪些:

目前最流行的Servlet容器软件包括: Tomcat、Jetty、Jboss等。


Tomcat

Tomcat和IIS等Web服务器一样,具有处理HTML页面的功能,另外它还是一个Servlet和JSP容器,独立的Servlet容器是Tomcat的默认模式。不过,Tomcat处理静态HTML的能力不如Apache服务器。


Jetty

Jetty 是一个开源的servlet容器,它为基于Java的web容器,例如JSP和servlet提供运行环境。Jetty是使用Java语言编写的,它的API以一组JAR包的形式发布。开发人员可以将Jetty容器实例化成一个对象,可以迅速为一些独立运行(stand-alone)的Java应用提供网络和web连接。


Jboss

Jboss是一个基于J2EE的开放源代码的应用服务器。 JBoss代码遵循LGPL许可,可以在任何商业应用中免费使用。JBoss是一个管理EJB的容器和服务器,支持EJB 1.1、EJB 2.0和EJB3的规范。但JBoss核心服务不包括支持servlet/JSP的WEB容器,一般与Tomcat或Jetty绑定使用。


Servlet是和平台无关的服务器端组件(java编写的,跨平台),它运行在Servlet容器中。

Servlet容器负责Servlet和客户的通信以及调用Servlet的方法,Servlet和客户的通信采用“请求/响应”的模式 Servlet可完成如下功能:

1、创建并返回基于客户请求的动态HTML页面

2、创建可嵌入到现有HTML 页面中的部分HTML 页面(HTML 片段)

3、与其它服务器资源(如数据库或基于Java的应用程序)进行通信


Servlet容器响应客户请求过程:

215229_9PD7_3839775


jsp作为Servlet技术的扩展,经常会有人将jsp和Servlet搞混。本文,将为大家带来servlet和jsp的区别,希望对大家有所帮助。


servlet和jsp的区别

1、Servlet在Java代码中可以通过HttpServletResponse对象动态输出HTML内容。

2、JSP是在静态HTML内容中嵌入Java代码,然后Java代码在被动态执行后生成HTML内容。


servlet和jsp各自的特点

1、Servlet虽然能够很好地组织业务逻辑代码,但是在Java源文件中,因为是通过字符串拼接的方式生成动态HTML内容,这样就容易导致代码维护困难、可读性差。

2、JSP虽然规避了Servlet在生成HTML内容方面的劣势,但是在HTML中混入大量、复杂的业务逻辑。


通过MVC双剑合璧

JSP和Servlet都有自身的适用环境,那么有没有什么办法能够让它们发挥各自的优势呢?答案是肯有的,MVC模式就能够完美解决这一问题。

MVC模式,是Model-View-Controller的简称,是软件工程中的一种软件架构模式,分为三个基本部分,分别是:模型(Model)、视图(View)和控制器(Controller):

Controller——负责转发请求,对请求进行处理

View——负责界面显示

Model——业务功能编写(例如算法实现)、数据库设计以及数据存取操作实现

在JSP/Servlet开发的软件系统中,这三个部分的描述如下所示:


229cf9ff5b1729eaf408fac56238eeb3_hd


1、Web浏览器发送HTTP请求到服务端,然后被Controller(Servlet)获取并进行处理(例如参数解析、请求转发)

2、Controller(Servlet)调用核心业务逻辑——Model部分,获得结果

3、Controller(Servlet)将逻辑处理结果交给View(JSP),动态输出HTML内容

4、动态生成的HTML内容返回到浏览器显示

MVC模式在Web开发中有很大的优势,它完美规避了JSP与Servlet各自的缺点,让Servlet只负责业务逻辑部分,而不会生成HTML代码;同时JSP中也不会充斥着大量的业务代码,这样能大提高了代码的可读性和可维护性。


相关阅读:

Servlet教程

JSP教程


Servlet 教程

Servlet 为创建基于 web 的应用程序提供了基于组件、独立于平台的方法,可以不受 CGI 程序的性能限制。Servlet 有权限访问所有的 Java API,包括访问企业级数据库的 JDBC API。
本教程将讲解如何使用 Java Servlet 来开发基于 web 的应用程序。

谁适合阅读本教程?

本教程是专为 Java 程序员设计的。在阅读本教程之前,需要先了解 Java Servlet 框架和它的 API。学习完本教程后,您会发现自己已经达到使用 Java Servlet 的中等水平,后续您可以自行通过更深入的学习和实践完成进阶。

阅读本教程前,您需要了解的知识:

在您开始阅读本教程之前,最好对 Java 编程语言有一个很好的理解。如果您对 web 应用程序和互联网如何工作的有基本的认识,将有助于您理解本教程。

Servlet 相关教程

Servlet 是什么?

Java Servlet 是运行在 Web 服务器或应用服务器上的程序,它是作为来自 Web 浏览器或其他 HTTP 客户端的请求和 HTTP 服务器上的数据库或应用程序之间的中间层。

使用 Servlet,您可以收集来自网页表单的用户输入,呈现来自数据库或者其他源的记录,还可以动态创建网页。

Java Servlet 通常情况下与使用 CGI(Common Gateway Interface,公共网关接口)实现的程序可以达到异曲同工的效果。但是相比于 CGI,Servlet 有以下几点优势:

  • 性能明显更好。
  • Servlet 在 Web 服务器的地址空间内执行。这样它就没有必要再创建一个单独的进程来处理每个客户端请求。
  • Servlet 是独立于平台的,因为它们是用 Java 编写的。
  • 服务器上的 Java 安全管理器执行了一系列限制,以保护服务器计算机上的资源。因此,Servlet 是可信的。
  • Java 类库的全部功能对 Servlet 来说都是可用的。它可以通过 sockets 和 RMI 机制与 applets、数据库或其他软件进行交互。

Servlet 架构

下图显示了 Servlet 在 Web 应用程序中的位置。

Servlet 架构

Servlet 任务

Servlet 执行以下主要任务:

  • 读取客户端(浏览器)发送的显式的数据。这包括网页上的 HTML 表单,或者也可以是来自 applet 或自定义的 HTTP 客户端程序的表单。
  • 读取客户端(浏览器)发送的隐式的 HTTP 请求数据。这包括 cookies、媒体类型和浏览器能理解的压缩格式等等。
  • 处理数据并生成结果。这个过程可能需要访问数据库,执行 RMI 或 CORBA 调用,调用 Web 服务,或者直接计算得出对应的响应。
  • 发送显式的数据(即文档)到客户端(浏览器)。该文档的格式可以是多种多样的,包括文本文件(HTML 或 XML)、二进制文件(GIF 图像)、Excel 等。
  • 发送隐式的 HTTP 响应到客户端(浏览器)。这包括告诉浏览器或其他客户端被返回的文档类型(例如 HTML),设置 cookies 和缓存参数,以及其他类似的任务。

Servlet 包

Java Servlet 是运行在带有支持 Java Servlet 规范的解释器的 web 服务器上的 Java 类。

Servlet 可以使用 javax.servletjavax.servlet.http 包创建,它是 Java 企业版的标准组成部分,Java 企业版是支持大型开发项目的 Java 类库的扩展版本。

这些类实现 Java Servlet 和 JSP 规范。在写本教程的时候,二者相应的版本分别是 Java Servlet 2.5 和 JSP 2.1。

Java Servlet 就像任何其他的 Java 类一样已经被创建和编译。在您安装 Servlet 包并把它们添加到您的计算机上的 Classpath 类路径中之后,您就可以通过 JDK 的 Java 编译器或任何其他编译器来编译 Servlet。

下一步呢?

接下来,本教程会带你一步一步地设置您的 Servlet 环境,以便开始后续的 Servlet 使用。因此,请系紧您的安全带,随我们一起开始 Servlet 的学习之旅吧!相信您会很喜欢这个教程的。

Servlet 环境设置

开发环境是您可以开发、测试、运行 Servlet 的地方。

就像任何其他的 Java 程序,您需要通过使用 Java 编译器 javac 编译 Servlet,在编译 Servlet 应用程序后,将它部署在配置的环境中以便测试和运行。

这个开发环境设置包括以下步骤:

设置 Java 开发工具包(Java Development Kit)

这一步涉及到下载 Java 软件开发工具包(SDK,即 Software Development Kit),并适当地设置 PATH 环境变量。

您可以从 Oracle 的 Java 网站下载 SDK:Java SE Downloads

一旦您下载了 SDK,请按照给定的指令来安装和配置设置。最后,设置 PATH 和 JAVA_HOME 环境变量指向包含 java 和 javac 的目录,通常分别为 java_install_dir/bin 和 java_install_dir。

如果您运行的是 Windows,并把 SDK 安装在 C:jdk1.5.0_20 中,则需要在您的 C:autoexec.bat 文件中放入下列的行:

set PATH=C:jdk1.5.0_20in;%PATH%set JAVA_HOME=C:jdk1.5.0_20

或者,在 Windows NT/2000/XP 中,您也可以用鼠标右键单击"我的电脑",选择"属性",再选择"高级","环境变量"。然后,更新 PATH 的值,按下"确定"按钮。

在 Unix(Solaris、Linux 等)上,如果 SDK 安装在 /usr/local/jdk1.5.0_20 中,并且您使用的是 C shell,则需要在您的 .cshrc 文件中放入下列的行:

setenv PATH /usr/local/jdk1.5.0_20/bin:$PATHsetenv JAVA_HOME /usr/local/jdk1.5.0_20

另外,如果您使用集成开发环境(IDE,即 Integrated Development Environment),比如 Borland JBuilder、Eclipse、IntelliJ IDEA 或 Sun ONE Studio,编译并运行一个简单的程序,以确认该 IDE 知道您安装的 Java 路径。

设置 Web 服务器:Tomcat

在市场上有许多 Web 服务器支持 Servlet。有些 Web 服务器是免费下载的,Tomcat 就是其中的一个。

Apache Tomcat 是一款 Java Servlet 和 JavaServer Pages 技术的开源软件实现,可以作为测试 Servlet 的独立服务器,而且可以集成到 Apache Web 服务器。下面是在电脑上安装 Tomcat 的步骤:

  • http://tomcat.apache.org/ 上下载最新版本的 Tomcat。
  • 一旦您下载了 Tomcat,解压缩到一个方便的位置。例如,如果您使用的是 Windows,则解压缩到 C:apache-tomcat-5.5.29 中,如果您使用的是 Linux/Unix,则解压缩到 /usr/local/apache-tomcat-5.5.29 中,并创建 CATALINA_HOME 环境变量指向这些位置。

在 Windows 上,可以通过执行下面的命令来启动 Tomcat:

 %CATALINA_HOME%instartup.bat or C:apache-tomcat-5.5.29instartup.bat

在 Unix(Solaris、Linux 等) 上,可以通过执行下面的命令来启动 Tomcat:

$CATALINA_HOME/bin/startup.shor/usr/local/apache-tomcat-5.5.29/bin/startup.sh

Tomcat 启动后,可以通过在浏览器地址栏输入 http://localhost:8080/ 访问 Tomcat 中的默认应用程序。如果一切顺利,那么会显示以下结果:

Tomcat Home page

有关配置和运行 Tomcat 的进一步信息可以查阅应用程序安装的文档,或者可以访问 Tomcat 网站:http://tomcat.apache.org

在 Windows 上,可以通过执行下面的命令来停止 Tomcat:

C:apache-tomcat-5.5.29inshutdown

在 Unix(Solaris、Linux 等) 上,可以通过执行下面的命令来停止 Tomcat:

/usr/local/apache-tomcat-5.5.29/bin/shutdown.sh

设置 CLASSPATH

由于 Servlet 不是 Java 平台标准版的组成部分,所以您必须为编译器指定 Servlet 类的路径。

如果您运行的是 Windows,则需要在您的 C:autoexec.bat 文件中放入下列的行:

set CATALINA=C:apache-tomcat-5.5.29set CLASSPATH=%CATALINA%commonlibservlet-api.jar;%CLASSPATH%

或者,在 Windows NT/2000/XP 中,您也可以用鼠标右键单击"我的电脑",选择"属性",再选择"高级","环境变量"。然后,更新 CLASSPATH 的值,按下"确定"按钮。

在 Unix(Solaris、Linux 等)上,如果您使用的是 C shell,则需要在您的 .cshrc 文件中放入下列的行:

setenv CATALINA=/usr/local/apache-tomcat-5.5.29setenv CLASSPATH $CATALINA/common/lib/servlet-api.jar:$CLASSPATH

注意:假设您的开发目录是 C:ServletDevel(在 Windows 上)或 /user/ServletDevel(在 UNIX 上),那么您还需要在 CLASSPATH 中添加这些目录,添加方式与上面的添加方式类似。

Servlet 生命周期

Servlet 生命周期可被定义为从创建直到毁灭的整个过程。以下是 Servlet 遵循的过程:

  • Servlet 通过调用 init () 方法进行初始化。
  • Servlet 调用 service() 方法来处理客户端的请求。
  • Servlet 通过调用 destroy() 方法终止(结束)。
  • 最后,Servlet 是由 JVM 的垃圾回收器进行垃圾回收的。

现在让我们详细讨论生命周期的方法。

init() 方法

init 方法被设计成只调用一次。它在第一次创建 Servlet 时被调用,在后续每次用户请求时不再调用。因此,它是用于一次性初始化,就像 Applet 的 init 方法一样。

Servlet 创建于用户第一次调用对应于该 Servlet 的 URL 时,但是您也可以指定 Servlet 在服务器第一次启动时被加载。

当用户调用一个 Servlet 时,就会创建一个 Servlet 实例,每一个用户请求都会产生一个新的线程,适当的时候移交给 doGet 或 doPost 方法。init() 方法简单地创建或加载一些数据,这些数据将被用于 Servlet 的整个生命周期。

init 方法的定义如下:

public void init() throws ServletException {  // 初始化代码...}

service() 方法

service() 方法是执行实际任务的主要方法。Servlet 容器(即 Web 服务器)调用 service() 方法来处理来自客户端(浏览器)的请求,并把格式化的响应写回给客户端。

每次服务器接收到一个 Servlet 请求时,服务器会产生一个新的线程并调用服务。service() 方法检查 HTTP 请求类型(GET、POST、PUT、DELETE 等),并在适当的时候调用 doGet、doPost、doPut,doDelete 等方法。

下面是该方法的特征:

public void service(ServletRequest request,                     ServletResponse response)       throws ServletException, IOException{}

service() 方法由容器调用,service 方法在适当的时候调用 doGet、doPost、doPut、doDelete 等方法。所以,您不用对 service() 方法做任何动作,您只需要根据来自客户端的请求类型来重载 doGet() 或 doPost() 即可。

doGet() 和 doPost() 方法是每次服务请求中最常用的方法。下面是这两种方法的特征。

doGet() 方法

GET 请求来自于一个 URL 的正常请求,或者来自于一个未指定 METHOD 的 HTML 表单,它由 doGet() 方法处理。

public void doGet(HttpServletRequest request,                  HttpServletResponse response)    throws ServletException, IOException {    // Servlet 代码}

doPost() 方法

POST 请求来自于一个特别指定了 METHOD 为 POST 的 HTML 表单,它由 doPost() 方法处理。

public void doPost(HttpServletRequest request,                   HttpServletResponse response)    throws ServletException, IOException {    // Servlet 代码}

destroy() 方法

destroy() 方法只会被调用一次,在 Servlet 生命周期结束时被调用。destroy() 方法可以让您的 Servlet 关闭数据库连接、停止后台线程、把 Cookie 列表或点击计数器写入到磁盘,并执行其他类似的清理活动。

在调用 destroy() 方法之后,servlet 对象被标记为垃圾回收。destroy 方法定义如下所示:

  public void destroy() {    // 终止化代码...  }

架构图

下图显示了一个典型的 Servlet 生命周期方案。

  • 第一个到达服务器的 HTTP 请求被委派到 Servlet 容器。
  • Servlet 容器在调用 service() 方法之前加载 Servlet。
  • 然后 Servlet 容器处理由多个线程产生的多个请求,每个线程执行一个单一的 Servlet 实例的 service() 方法。

Servlet 生命周期 

以下为生命周期的一个简单实例:

//servlet生命周期,的三个方法,1.被创建,执行且只执行一次init方法,2.提供服务,执行service方法,执行多次 3.被销毁,当Servlet服务器正常关闭时,执行destroy方法,只执行一次。          @Override     public void init() throws ServletException {         // TODO Auto-generated method stub         super.init();     }          @Override     protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {         // TODO Auto-generated method stub         super.service(req, resp);     }          @Override     public void destroy() {         // TODO Auto-generated method stub         super.destroy();     }


Servlet 实例

Servlet 是服务 HTTP 请求并实现 javax.servlet.Servlet 接口的 Java 类。Web 应用程序开发人员通常编写 Servlet 来扩展 javax.servlet.http.HttpServlet,并实现 Servlet 接口的抽象类专门用来处理 HTTP 请求。

Hello World 示例代码

下面是 Servlet 输出 Hello World 的示例源代码:

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*;// 扩展 HttpServlet 类public class HelloWorld extends HttpServlet {   private String message;  public void init() throws ServletException  {      // 执行必需的初始化      message = "Hello World";  }  public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 设置响应内容类型      response.setContentType("text/html");      // 实际的逻辑是在这里      PrintWriter out = response.getWriter();      out.println("<h1>" + message + "</h1>");  }    public void destroy()  {      // 什么也不做  }}

编译 Servlet

让我们把上面的代码写在 HelloWorld.java 文件中,把这个文件放在 C:ServletDevel(在 Windows 上)或 /usr/ServletDevel(在 UNIX 上)中,您还需要把这些目录添加到 CLASSPATH 中。

假设您的环境已经正确地设置,进入 ServletDevel 目录,并编译 HelloWorld.java,如下所示:

$ javac HelloWorld.java

如果 Servlet 依赖于任何其他库,您必须在 CLASSPATH 中包含那些 JAR 文件。在这里,我只包含了 servlet-api.jar JAR 文件,因为我没有在 Hello World 程序中使用任何其他库。

该命令行使用 Sun Microsystems Java 软件开发工具包(JDK)内置的 javac 编译器。为使该命令正常工作,您必须 PATH 环境变量中使用的 Java SDK 的位置。

如果一切顺利,上面编译会在同一目录下生成 HelloWorld.class 文件。下一节将讲解已编译的 Servlet 如何部署在生产中。

Servlet 部署

默认情况下,Servlet 应用程序位于路径 <Tomcat-installation-directory>/webapps/ROOT 下,且类文件放在 <Tomcat-installation-directory>/webapps/ROOT/WEB-INF/classes 中。

如果您有一个完全合格的类名称 com.myorg.MyServlet,那么这个 Servlet 类必须位于 WEB-INF/classes/com/myorg/MyServlet.class 中。

现在,让我们把 HelloWorld.class 复制到 <Tomcat-installation-directory>/webapps/ROOT/WEB-INF/classes 中,并在位于 <Tomcat-installation-directory>/webapps/ROOT/WEB-INF/ 的 web.xml 文件中创建以下条目:

    <servlet>        <servlet-name>HelloWorld</servlet-name>        <servlet-class>HelloWorld</servlet-class>    </servlet>    <servlet-mapping>        <servlet-name>HelloWorld</servlet-name>        <url-pattern>/HelloWorld</url-pattern>    </servlet-mapping>

上面的条目要被创建在 web.xml 文件中的 <web-app>...</web-app> 标签内。在该文件中可能已经有各种可用的条目,但不要在意。

到这里,您基本上已经完成了,现在让我们使用 <Tomcat-installation-directory>instartup.bat(在 Windows 上)或 <Tomcat-installation-directory>/bin/startup.sh(在 Linux/Solaris 等上)启动 tomcat 服务器,最后在浏览器的地址栏中输入 http://localhost:8080/HelloWorld。如果一切顺利,您会看到下面的结果:

Servlet 实例

Servlet 表单数据

很多情况下,需要传递一些信息,从浏览器到 Web 服务器,最终到后台程序。浏览器使用两种方法可将这些信息传递到 Web 服务器,分别为 GET 方法和 POST 方法。

GET 方法

GET 方法向页面请求发送已编码的用户信息。页面和已编码的信息中间用 ? 字符分隔,如下所示:

http://www.test.com/hello?key1=value1&key2=value2

GET 方法是默认的从浏览器向 Web 服务器传递信息的方法,它会产生一个很长的字符串,出现在浏览器的地址栏中。如果您要向服务器传递的是密码或其他的敏感信息,请不要使用 GET 方法。GET 方法有大小限制:请求字符串中最多只能有 1024 个字符。

这些信息使用 QUERY_STRING 头传递,并可以通过 QUERY_STRING 环境变量访问,Servlet 使用 doGet() 方法处理这种类型的请求。

POST 方法

另一个向后台程序传递信息的比较可靠的方法是 POST 方法。POST 方法打包信息的方式与 GET 方法基本相同,但是 POST 方法不是把信息作为 URL 中 ? 字符后的文本字符串进行发送,而是把这些信息作为一个单独的消息。消息以标准输出的形式传到后台程序,您可以解析和使用这些标准输出。Servlet 使用 doPost() 方法处理这种类型的请求。

使用 Servlet 读取表单数据

Servlet 处理表单数据,这些数据会根据不同的情况使用不同的方法自动解析:

  • getParameter():您可以调用 request.getParameter() 方法来获取表单参数的值。
  • getParameterValues():如果参数出现一次以上,则调用该方法,并返回多个值,例如复选框。
  • getParameterNames():如果您想要得到当前请求中的所有参数的完整列表,则调用该方法。

使用 URL 的 GET 方法实例

下面是一个简单的 URL,将使用 GET 方法向 HelloForm 程序传递两个值。

http://localhost:8080/HelloForm?first_name=ZARA&last_name=ALI

下面是处理 Web 浏览器输入的 HelloForm.java Servlet 程序。我们将使用 getParameter() 方法,可以很容易地访问传递的信息:

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*;// 扩展 HttpServlet 类public class HelloForm extends HttpServlet {   public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 设置响应内容类型      response.setContentType("text/html");      PrintWriter out = response.getWriter();      String title = "使用 GET 方法读取表单数据";      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +                "<html>
" +                "<head><title>" + title + "</title></head>
" +                "<body bgcolor="#f0f0f0">
" +                "<h1 align="center">" + title + "</h1>
" +                "<ul>
" +                "  <li><b>名字</b>:"                + request.getParameter("first_name") + "
" +                "  <li><b>姓氏</b>:"                + request.getParameter("last_name") + "
" +                "</ul>
" +                "</body></html>");  }}

假设您的环境已经正确地设置,编译 HelloForm.java,如下所示:

$ javac HelloForm.java

如果一切顺利,上述编译会产生 HelloForm.class 文件。接下来,您就必须把该类文件复制到 <Tomcat-installation-directory>/webapps/ROOT/WEB-INF/classes 中,并在位于 <Tomcat-installation-directory>/webapps/ROOT/WEB-INF/ 的 web.xml 文件中创建以下条目:

    <servlet>        <servlet-name>HelloForm</servlet-name>        <servlet-class>HelloForm</servlet-class>    </servlet>    <servlet-mapping>        <servlet-name>HelloForm</servlet-name>        <url-pattern>/HelloForm</url-pattern>    </servlet-mapping>

现在在浏览器的地址栏中输入 http://localhost:8080/HelloForm?first_name=ZARA&last_name=ALI ,并在触发上述命令之前确保已经启动 Tomcat 服务器。如果一切顺利,您会得到下面的结果:

使用 GET 方法读取表单数据

  • 名字:ZARA
  • 姓氏:ALI

使用表单的 GET 方法实例

下面是一个简单的实例,使用 HTML 表单和提交按钮传递两个值。我们将使用相同的 Servlet HelloForm 来处理输入。

<html><body><form action="HelloForm" method="GET">名字:<input type="text" name="first_name"><br />姓氏:<input type="text" name="last_name" /><input type="submit" value="提交" /></form></body></html>

保存这个 HTML 到 hello.htm 文件中,并把它放在 <Tomcat-installation-directory>/webapps/ROOT 目录下。当您访问 http://localhost:8080/Hello.htm 时,下面是上面表单的实际输出。


尝试输入名字和姓氏,然后点击"提交"按钮,在您本机上查看输出结果。基于所提供的输入,它会产生与上一个实例类似的结果。

使用表单的 POST 方法实例

让我们对上面的 Servlet 做小小的修改,以便它可以处理 GET 和 POST 方法。下面的 HelloForm.java Servlet 程序使用 GET 和 POST 方法处理由 Web 浏览器给出的输入。

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*;// 扩展 HttpServlet 类public class HelloForm extends HttpServlet {   // 处理 GET 方法请求的方法  public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 设置响应内容类型      response.setContentType("text/html");      PrintWriter out = response.getWriter();      String title = "Using GET Method to Read Form Data";      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +                "<html>
" +                "<head><title>" + title + "</title></head>
" +                "<body bgcolor="#f0f0f0">
" +                "<h1 align="center">" + title + "</h1>
" +                "<ul>
" +                "  <li><b>名字</b>:"                + request.getParameter("first_name") + "
" +                "  <li><b>姓氏</b>:"                + request.getParameter("last_name") + "
" +                "</ul>
" +                "</body></html>");  }  // 处理 POST 方法请求的方法  public void doPost(HttpServletRequest request,                     HttpServletResponse response)      throws ServletException, IOException {     doGet(request, response);  }}

现在,编译部署上述的 Servlet,并使用带有 POST 方法的 Hello.htm 进行测试,如下所示:

<html><body><form action="HelloForm" method="POST">名字:<input type="text" name="first_name"><br />姓氏:<input type="text" name="last_name" /><input type="submit" value="提交" /></form></body></html>

下面是上面表单的实际输出,尝试输入名字和姓氏,然后点击"提交"按钮,在您本机上查看输出结果。


基于所提供的输入,它会产生与上一个实例类似的结果。

将复选框数据传递到 Servlet 程序

当需要选择一个以上的选项时,则使用复选框。

下面是一个 HTML 代码实例 CheckBox.htm,一个带有两个复选框的表单。

<html><body><form action="CheckBox" method="POST" target="_blank"><input type="checkbox" name="maths" checked="checked" /> 数学<input type="checkbox" name="physics"  /> 物理<input type="checkbox" name="chemistry" checked="checked" />                                                 化学<input type="submit" value="选择学科" /></form></body></html>

这段代码的结果是下面的表单:


下面是 CheckBox.java Servlet 程序,处理 Web 浏览器给出的复选框输入。

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*;// 扩展 HttpServlet 类public class CheckBox extends HttpServlet {   // 处理 GET 方法请求的方法  public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 设置响应内容类型      response.setContentType("text/html");      PrintWriter out = response.getWriter();     String title = "读取复选框数据";      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +                "<html>
" +                "<head><title>" + title + "</title></head>
" +                "<body bgcolor="#f0f0f0">
" +                "<h1 align="center">" + title + "</h1>
" +                "<ul>
" +                "  <li><b>数学标识:</b>: "                + request.getParameter("maths") + "
" +                "  <li><b>物理标识:</b>: "                + request.getParameter("physics") + "
" +                "  <li><b>化学标识:</b>: "                + request.getParameter("chemistry") + "
" +                "</ul>
" +                "</body></html>");  }  // 处理 POST 方法请求的方法  public void doPost(HttpServletRequest request,                     HttpServletResponse response)      throws ServletException, IOException {     doGet(request, response);  }}

上面的实例将显示下面的结果:

读取复选框数据

  • 数学标识:on
  • 物理标识:null
  • 化学标识:on

读取所有的表单参数

以下是通用的实例,使用 HttpServletRequest 的 getParameterNames() 方法读取所有可用的表单参数。该方法返回一个枚举,其中包含未指定顺序的参数名。

一旦我们有一个枚举,我们可以以标准方式循环枚举,使用 hasMoreElements() 方法来确定何时停止,使用 nextElement() 方法来获取每个参数的名称。

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*;import java.util.*;// 扩展 HttpServlet 类public class ReadParams extends HttpServlet {   // 处理 GET 方法请求的方法  public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 设置响应内容类型      response.setContentType("text/html");      PrintWriter out = response.getWriter();     String title = "读取所有的表单数据";      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +        "<html>
" +        "<head><title>" + title + "</title></head>
" +        "<body bgcolor="#f0f0f0">
" +        "<h1 align="center">" + title + "</h1>
" +        "<table width="100%" border="1" align="center">
" +        "<tr bgcolor="#949494">
" +        "<th>参数名称</th><th>参数值</th>
"+        "</tr>
");      Enumeration paramNames = request.getParameterNames();            while(paramNames.hasMoreElements()) {         String paramName = (String)paramNames.nextElement();         out.print("<tr><td>" + paramName + "</td>
<td>");         String[] paramValues =                request.getParameterValues(paramName);         // 读取单个值的数据         if (paramValues.length == 1) {           String paramValue = paramValues[0];           if (paramValue.length() == 0)             out.println("<i>No Value</i>");           else             out.println(paramValue);         } else {             // 读取多个值的数据             out.println("<ul>");             for(int i=0; i < paramValues.length; i++) {                 out.println("<li>" + paramValues[i]);             }             out.println("</ul>");         }      }      out.println("</tr>
</table>
</body></html>");  }  // 处理 POST 方法请求的方法  public void doPost(HttpServletRequest request,                     HttpServletResponse response)      throws ServletException, IOException {     doGet(request, response);  }}

现在,通过下面的表单尝试上面的 Servlet:

<html><body><form action="ReadParams" method="POST" target="_blank"><input type="checkbox" name="maths" checked="checked" /> 数学<input type="checkbox" name="physics"  /> 物理<input type="checkbox" name="chemistry" checked="checked" /> 化学<input type="submit" value="选择学科" /></form></body></html>

现在使用上面的表单调用 Servlet,将产生以下结果:

读取所有的表单数据

参数名称参数值
maths on
chemistry on

您可以尝试使用上面的 Servlet 来读取其他的表单数据,比如文本框、单选按钮或下拉框等。

Servlet 客户端 HTTP 请求

当浏览器请求网页时,它会向 Web 服务器发送特定信息,这些信息不能被直接读取,因为这些信息是作为 HTTP 请求的头的一部分进行传输的。您可以查看 HTTP 协议 了解更多相关信息。

以下是来自于浏览器端的重要头信息,您可以在 Web 编程中频繁使用:

头信息描述
Accept这个头信息指定浏览器或其他客户端可以处理的 MIME 类型。值 image/pngimage/jpeg 是最常见的两种可能值。
Accept-Charset这个头信息指定浏览器可以用来显示信息的字符集。例如 ISO-8859-1。
Accept-Encoding这个头信息指定浏览器知道如何处理的编码类型。值 gzipcompress 是最常见的两种可能值。
Accept-Language这个头信息指定客户端的首选语言,在这种情况下,Servlet 会产生多种语言的结果。例如,en、en-us、ru 等。
Authorization这个头信息用于客户端在访问受密码保护的网页时识别自己的身份。
Connection这个头信息指示客户端是否可以处理持久 HTTP 连接。持久连接允许客户端或其他浏览器通过单个请求来检索多个文件。值 Keep-Alive 意味着使用了持续连接。
Content-Length这个头信息只适用于 POST 请求,并给出 POST 数据的大小(以字节为单位)。
Cookie这个头信息把之前发送到浏览器的 cookies 返回到服务器。
Host这个头信息指定原始的 URL 中的主机和端口。
If-Modified-Since这个头信息表示只有当页面在指定的日期后已更改时,客户端想要的页面。如果没有新的结果可以使用,服务器会发送一个 304 代码,表示 Not Modified 头信息。
If-Unmodified-Since这个头信息是 If-Modified-Since 的对立面,它指定只有当文档早于指定日期时,操作才会成功。
Referer这个头信息指示所指向的 Web 页的 URL。例如,如果您在网页 1,点击一个链接到网页 2,当浏览器请求网页 2 时,网页 1 的 URL 就会包含在 Referer 头信息中。
User-Agent这个头信息识别发出请求的浏览器或其他客户端,并可以向不同类型的浏览器返回不同的内容。

读取 HTTP 头的方法

下面的方法可用在 Servlet 程序中读取 HTTP 头。这些方法通过 HttpServletRequest 对象可用。

序号方法 & 描述
1Cookie[] getCookies()
返回一个数组,包含客户端发送该请求的所有的 Cookie 对象。
2Enumeration getAttributeNames()
返回一个枚举,包含提供给该请求可用的属性名称。
3Enumeration getHeaderNames()
返回一个枚举,包含在该请求中包含的所有的头名。
4Enumeration getParameterNames()
返回一个 String 对象的枚举,包含在该请求中包含的参数的名称。
5HttpSession getSession()
返回与该请求关联的当前 session 会话,或者如果请求没有 session 会话,则创建一个。
6HttpSession getSession(boolean create)
返回与该请求关联的当前 HttpSession,或者如果没有当前会话,且创建是真的,则返回一个新的 session 会话。
7Locale getLocale()
基于 Accept-Language 头,返回客户端接受内容的首选的区域设置。
8Object getAttribute(String name)
以对象形式返回已命名属性的值,如果没有给定名称的属性存在,则返回 null。
9ServletInputStream getInputStream()
使用 ServletInputStream,以二进制数据形式检索请求的主体。
10String getAuthType()
返回用于保护 Servlet 的身份验证方案的名称,例如,"BASIC" 或 "SSL",如果JSP没有受到保护则返回 null。
11String getCharacterEncoding()
返回请求主体中使用的字符编码的名称。
12String getContentType()
返回请求主体的 MIME 类型,如果不知道类型则返回 null。
13String getContextPath()
返回指示请求上下文的请求 URI 部分。
14String getHeader(String name)
以字符串形式返回指定的请求头的值。
15String getMethod()
返回请求的 HTTP 方法的名称,例如,GET、POST 或 PUT。
16String getParameter(String name)
以字符串形式返回请求参数的值,或者如果参数不存在则返回 null。
17String getPathInfo()
当请求发出时,返回与客户端发送的 URL 相关的任何额外的路径信息。
18String getProtocol()
返回请求协议的名称和版本。
19String getQueryString()
返回包含在路径后的请求 URL 中的查询字符串。
20String getRemoteAddr()
返回发送请求的客户端的互联网协议(IP)地址。
21String getRemoteHost()
返回发送请求的客户端的完全限定名称。
22String getRemoteUser()
如果用户已通过身份验证,则返回发出请求的登录用户,或者如果用户未通过身份验证,则返回 null。
23String getRequestURI()
从协议名称直到 HTTP 请求的第一行的查询字符串中,返回该请求的 URL 的一部分。
24String getRequestedSessionId()
返回由客户端指定的 session 会话 ID。
25String getServletPath()
返回调用 JSP 的请求的 URL 的一部分。
26String[] getParameterValues(String name)
返回一个字符串对象的数组,包含所有给定的请求参数的值,如果参数不存在则返回 null。
27boolean isSecure()
返回一个布尔值,指示请求是否使用安全通道,如 HTTPS。
28int getContentLength()
以字节为单位返回请求主体的长度,并提供输入流,或者如果长度未知则返回 -1。
29int getIntHeader(String name)
返回指定的请求头的值为一个 int 值。
30int getServerPort()
返回接收到这个请求的端口号。

HTTP Header 请求实例

下面的实例使用 HttpServletRequest 的 getHeaderNames() 方法读取 HTTP 头信息。该方法返回一个枚举,包含与当前的 HTTP 请求相关的头信息。

一旦我们有一个枚举,我们可以以标准方式循环枚举,使用 hasMoreElements() 方法来确定何时停止,使用 nextElement() 方法来获取每个参数的名称。

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*;import java.util.*; // 扩展 HttpServlet 类public class DisplayHeader extends HttpServlet {   // 处理 GET 方法请求的方法  public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 设置响应内容类型      response.setContentType("text/html");       PrintWriter out = response.getWriter();    String title = "HTTP Header 请求实例";      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +        "<html>
" +        "<head><title>" + title + "</title></head>
"+        "<body bgcolor="#f0f0f0">
" +        "<h1 align="center">" + title + "</h1>
" +        "<table width="100%" border="1" align="center">
" +        "<tr bgcolor="#949494">
" +        "<th>Header 名称</th><th>Header 值</th>
"+        "</tr>
");       Enumeration headerNames = request.getHeaderNames();            while(headerNames.hasMoreElements()) {         String paramName = (String)headerNames.nextElement();         out.print("<tr><td>" + paramName + "</td>
");         String paramValue = request.getHeader(paramName);         out.println("<td> " + paramValue + "</td></tr>
");      }      out.println("</table>
</body></html>");  }  // 处理 POST 方法请求的方法  public void doPost(HttpServletRequest request,                     HttpServletResponse response)      throws ServletException, IOException {     doGet(request, response);  }}

现在,调用上面的 Servlet 会产生以下结果:

HTTP Header 请求实例

Header 名称Header 值
accept */*
accept-language en-us
user-agent Mozilla/4.0 (compatible; MSIE 7.0; Windows NT 5.1; Trident/4.0; InfoPath.2; MS-RTC LM 8)
accept-encoding gzip, deflate
host localhost:8080
connection Keep-Alive
cache-control no-cache

Servlet 服务器 HTTP 响应

正如前面的章节中讨论的那样,当一个 Web 服务器响应一个 HTTP 请求时,响应通常包括一个状态行、一些响应报头、一个空行和文档。一个典型的响应如下所示:

HTTP/1.1 200 OKContent-Type: text/htmlHeader2: ......HeaderN: ...  (Blank Line)<!doctype ...><html><head>...</head><body>...</body></html>

状态行包括 HTTP 版本(在本例中为 HTTP/1.1)、一个状态码(在本例中为 200)和一个对应于状态码的短消息(在本例中为 OK)。

下表总结了从 Web 服务器端返回到浏览器的最有用的 HTTP 1.1 响应报头,您会在 Web 编程中频繁地使用它们:

头信息描述
Allow这个头信息指定服务器支持的请求方法(GET、POST 等)。
Cache-Control这个头信息指定响应文档在何种情况下可以安全地缓存。可能的值有:public、privateno-cache 等。Public 意味着文档是可缓存,Private 意味着文档是单个用户私用文档,且只能存储在私有(非共享)缓存中,no-cache 意味着文档不应被缓存。
Connection这个头信息指示浏览器是否使用持久 HTTP 连接。值 close 指示浏览器不使用持久 HTTP 连接,值 keep-alive 意味着使用持久连接。
Content-Disposition这个头信息可以让您请求浏览器要求用户以给定名称的文件把响应保存到磁盘。
Content-Encoding在传输过程中,这个头信息指定页面的编码方式。
Content-Language这个头信息表示文档编写所使用的语言。例如,en、en-us、ru 等。
Content-Length这个头信息指示响应中的字节数。只有当浏览器使用持久(keep-alive)HTTP 连接时才需要这些信息。
Content-Type这个头信息提供了响应文档的 MIME(Multipurpose Internet Mail Extension)类型。
Expires这个头信息指定内容过期的时间,在这之后内容不再被缓存。
Last-Modified这个头信息指示文档的最后修改时间。然后,客户端可以缓存文件,并在以后的请求中通过 If-Modified-Since 请求头信息提供一个日期。
Location这个头信息应被包含在所有的带有状态码的响应中。在 300s 内,这会通知浏览器文档的地址。浏览器会自动重新连接到这个位置,并获取新的文档。
Refresh这个头信息指定浏览器应该如何尽快请求更新的页面。您可以指定页面刷新的秒数。
Retry-After这个头信息可以与 503(Service Unavailable 服务不可用)响应配合使用,这会告诉客户端多久就可以重复它的请求。
Set-Cookie这个头信息指定一个与页面关联的 cookie。

设置 HTTP 响应报头的方法

下面的方法可用于在 Servlet 程序中设置 HTTP 响应报头。这些方法通过 HttpServletResponse 对象可用。

序号方法 & 描述
1String encodeRedirectURL(String url)
为 sendRedirect 方法中使用的指定的 URL 进行编码,或者如果编码不是必需的,则返回 URL 未改变。
2String encodeURL(String url)
对包含 session 会话 ID 的指定 URL 进行编码,或者如果编码不是必需的,则返回 URL 未改变。
3boolean containsHeader(String name)
返回一个布尔值,指示是否已经设置已命名的响应报头。
4boolean isCommitted()
返回一个布尔值,指示响应是否已经提交。
5void addCookie(Cookie cookie)
把指定的 cookie 添加到响应。
6void addDateHeader(String name, long date)
添加一个带有给定的名称和日期值的响应报头。
7void addHeader(String name, String value)
添加一个带有给定的名称和值的响应报头。
8void addIntHeader(String name, int value)
添加一个带有给定的名称和整数值的响应报头。
9void flushBuffer()
强制任何在缓冲区中的内容被写入到客户端。
10void reset()
清除缓冲区中存在的任何数据,包括状态码和头。
11void resetBuffer()
清除响应中基础缓冲区的内容,不清除状态码和头。
12void sendError(int sc)
使用指定的状态码发送错误响应到客户端,并清除缓冲区。
13void sendError(int sc, String msg)
使用指定的状态发送错误响应到客户端。
14void sendRedirect(String location)
使用指定的重定向位置 URL 发送临时重定向响应到客户端。
15void setBufferSize(int size)
为响应主体设置首选的缓冲区大小。
16void setCharacterEncoding(String charset)
设置被发送到客户端的响应的字符编码(MIME 字符集)例如,UTF-8。
17void setContentLength(int len)
设置在 HTTP Servlet 响应中的内容主体的长度,该方法设置 HTTP Content-Length 头。
18void setContentType(String type)
如果响应还未被提交,设置被发送到客户端的响应的内容类型。
19void setDateHeader(String name, long date)
设置一个带有给定的名称和日期值的响应报头。
20void setHeader(String name, String value)
设置一个带有给定的名称和值的响应报头。
21void setIntHeader(String name, int value)
设置一个带有给定的名称和整数值的响应报头。
22void setLocale(Locale loc)
如果响应还未被提交,设置响应的区域。
23void setStatus(int sc)
为该响应设置状态码。

HTTP Header 响应实例

您已经在前面的实例中看到 setContentType() 方法,下面的实例也使用了同样的方法,此外,我们会用 setIntHeader() 方法来设置 Refresh 头。

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*;import java.util.*; // 扩展 HttpServlet 类public class Refresh extends HttpServlet {   // 处理 GET 方法请求的方法  public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 设置刷新自动加载时间为 5 秒      response.setIntHeader("Refresh", 5);       // 设置响应内容类型      response.setContentType("text/html");       // Get current time      Calendar calendar = new GregorianCalendar();      String am_pm;      int hour = calendar.get(Calendar.HOUR);      int minute = calendar.get(Calendar.MINUTE);      int second = calendar.get(Calendar.SECOND);      if(calendar.get(Calendar.AM_PM) == 0)        am_pm = "AM";      else        am_pm = "PM";       String CT = hour+":"+ minute +":"+ second +" "+ am_pm;          PrintWriter out = response.getWriter();      String title = "自动刷新 Header 设置";      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +        "<html>
" +        "<head><title>" + title + "</title></head>
"+        "<body bgcolor="#f0f0f0">
" +        "<h1 align="center">" + title + "</h1>
" +        "<p>当前时间是:" + CT + "</p>
");  }  // 处理 POST 方法请求的方法  public void doPost(HttpServletRequest request,                     HttpServletResponse response)      throws ServletException, IOException {     doGet(request, response);  }}

现在,调用上面的 Servlet,每隔 5 秒会显示当前系统时间。只要运行 Servlet 并稍等片刻,即可看到如下的结果:

自动刷新 Header 设置

当前时间是:9:44:50 PM

Servlet HTTP 状态码

HTTP 请求和 HTTP 响应消息的格式是类似的,结构如下:

  • 初始状态行 + 回车换行符(回车+换行)
  • 零个或多个标题行+回车换行符
  • 一个空白行,即回车换行符
  • 一个可选的消息主体,比如文件、查询数据或查询输出

例如,服务器的响应头如下所示:

HTTP/1.1 200 OKContent-Type: text/htmlHeader2: ......HeaderN: ...  (Blank Line)<!doctype ...><html><head>...</head><body>...</body></html>

状态行包括 HTTP 版本(在本例中为 HTTP/1.1)、一个状态码(在本例中为 200)和一个对应于状态码的短消息(在本例中为 OK)。

以下是可能从 Web 服务器返回的 HTTP 状态码和相关的信息列表:

代码 消息 描述
100 Continue 只有请求的一部分已经被服务器接收,但只要它没有被拒绝,客户端应继续该请求。
101 Switching Protocols 服务器切换协议。
200 OK 请求成功。
201 Created 该请求是完整的,并创建一个新的资源。
202 Accepted 该请求被接受处理,但是该处理是不完整的。
203 Non-authoritative Information  
204 No Content  
205 Reset Content  
206 Partial Content  
300 Multiple Choices 链接列表。用户可以选择一个链接,进入到该位置。最多五个地址。
301 Moved Permanently 所请求的页面已经转移到一个新的 URL。
302 Found 所请求的页面已经临时转移到一个新的 URL。
303 See Other 所请求的页面可以在另一个不同的 URL 下被找到。
304 Not Modified  
305 Use Proxy  
306 Unused 在以前的版本中使用该代码。现在已不再使用它,但代码仍被保留。
307 Temporary Redirect 所请求的页面已经临时转移到一个新的 URL。
400Bad Request 服务器不理解请求。
401 Unauthorized 所请求的页面需要用户名和密码。
402 Payment Required 您还不能使用该代码。
403 Forbidden 禁止访问所请求的页面。
404 Not Found 服务器无法找到所请求的页面。.
405 Method Not Allowed 在请求中指定的方法是不允许的。
406 Not Acceptable 服务器只生成一个不被客户端接受的响应。
407 Proxy Authentication Required 在请求送达之前,您必须使用代理服务器的验证。
408 Request Timeout 请求需要的时间比服务器能够等待的时间长,超时。
409 Conflict 请求因为冲突无法完成。
410 Gone 所请求的页面不再可用。
411 Length Required "Content-Length" 未定义。服务器无法处理客户端发送的不带 Content-Length 的请求信息。
412 Precondition Failed 请求中给出的先决条件被服务器评估为 false。
413 Request Entity Too Large 服务器不接受该请求,因为请求实体过大。
414 Request-url Too Long 服务器不接受该请求,因为 URL 太长。当您转换一个 "post" 请求为一个带有长的查询信息的 "get" 请求时发生。
415 Unsupported Media Type 服务器不接受该请求,因为媒体类型不被支持。
417 Expectation Failed  
500Internal Server Error 未完成的请求。服务器遇到了一个意外的情况。
501 Not Implemented 未完成的请求。服务器不支持所需的功能。
502 Bad Gateway 未完成的请求。服务器从上游服务器收到无效响应。
503 Service Unavailable 未完成的请求。服务器暂时超载或死机。
504 Gateway Timeout 网关超时。
505 HTTP Version Not Supported 服务器不支持"HTTP协议"版本。

设置 HTTP 状态代码的方法

下面的方法可用于在 Servlet 程序中设置 HTTP 状态码。这些方法通过 HttpServletResponse 对象可用。

序号方法 & 描述
1public void setStatus ( int statusCode )
该方法设置一个任意的状态码。setStatus 方法接受一个 int(状态码)作为参数。如果您的反应包含了一个特殊的状态码和文档,请确保在使用 PrintWriter 实际返回任何内容之前调用 setStatus。
2public void sendRedirect(String url)
该方法生成一个 302 响应,连同一个带有新文档 URL 的 Location 头。
3public void sendError(int code, String message)
该方法发送一个状态码(通常为 404),连同一个在 HTML 文档内部自动格式化并发送到客户端的短消息。

HTTP 状态码实例

下面的例子把 407 错误代码发送到客户端浏览器,浏览器会显示 "Need authentication!!!" 消息。

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*;import java.util.*;// 扩展 HttpServlet 类public class showError extends HttpServlet {   // 处理 GET 方法请求的方法  public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 设置错误代码和原因      response.sendError(407, "Need authentication!!!" );  }  // 处理 POST 方法请求的方法  public void doPost(HttpServletRequest request,                     HttpServletResponse response)      throws ServletException, IOException {     doGet(request, response);  }}

现在,调用上面的 Servlet 将显示以下结果:

HTTP Status 407 - Need authentication!!!

type Status report

message Need authentication!!!

description The client must first authenticate itself with the proxy (Need authentication!!!).

Apache Tomcat/5.5.29

Servlet 编写过滤器

Servlet 过滤器是可用于 Servlet 编程的 Java 类,有以下目的:

  • 在客户端的请求访问后端资源之前,拦截这些请求。
  • 在服务器的响应发送回客户端之前,处理这些响应。

根据规范建议的各种类型的过滤器:

  • 身份验证过滤器(Authentication Filters)。
  • 数据压缩过滤器(Data compression Filters)。
  • 加密过滤器(Encryption Filters)。
  • 触发资源访问事件过滤器。
  • 图像转换过滤器(Image Conversion Filters)。
  • 日志记录和审核过滤器(Logging and Auditing Filters)。
  • MIME-TYPE 链过滤器(MIME-TYPE Chain Filters)。
  • 标记化过滤器(Tokenizing Filters)。
  • XSL/T 过滤器(XSL/T Filters),转换 XML 内容。

过滤器被部署在部署描述符文件 web.xml 中,然后映射到您的应用程序的部署描述符中的 Servlet 名称或 URL 模式。

当 Web 容器启动 Web 应用程序时,它会为您在部署描述符中声明的每一个过滤器创建一个实例。该过滤器执行的顺序是按它们在部署描述符中声明的顺序。

Servlet 过滤器方法

过滤器是一个实现了 javax.servlet.Filter 接口的 Java 类。javax.servlet.Filter 接口定义了三个方法:

序号方法 & 描述
1public void doFilter (ServletRequest, ServletResponse, FilterChain)
该方法在每次一个请求/响应对因客户端在链的末端请求资源而通过链传递时由容器调用。
2public void init(FilterConfig filterConfig)
该方法由 Web 容器调用,指示一个过滤器被放入服务。
3public void destroy()
该方法由 Web 容器调用,指示一个过滤器被取出服务。

FilterConfig 使用

Filter 的 init 方法中提供了一个 FilterConfig 对象。

如 web.xml 文件配置如下:

<filter>	<filter-name>LoginFilter</filter-name>	<filter-class>com.51coolma.test.LogFilter</filter-class>	<init-param>		<param-name>Site</param-name>		<param-value>51coolma在线教程</param-value>	</init-param>	</filter>

在 init 方法使用 FilterConfig 对象获取参数:

public void  init(FilterConfig config) throws ServletException {	// 获取初始化参数	String site = config.getInitParameter("Site"); 	// 输出初始化参数	System.out.println("网站名称: " + site); }

Servlet 过滤器实例

以下是 Servlet 过滤器的实例,将输出客户端的 IP 地址和当前的日期时间。本实例让您对 Servlet 过滤器有基本的了解,您可以使用相同的概念编写更复杂的过滤器应用程序:

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*;import java.util.*;// 实现 Filter 类public class LogFilter implements Filter  {   public void  init(FilterConfig config)                          throws ServletException{      // 获取初始化参数      String testParam = config.getInitParameter("test-param");       // 输出初始化参数      System.out.println("Test Param: " + testParam);    }   public void  doFilter(ServletRequest request,                  ServletResponse response,                 FilterChain chain)                  throws java.io.IOException, ServletException {      // 获取客户机的 IP 地址         String ipAddress = request.getRemoteAddr();      // 记录 IP 地址和当前时间戳      System.out.println("IP "+ ipAddress + ", Time "                                       + new Date().toString());      // 把请求传回过滤链      chain.doFilter(request,response);   }   public void destroy( ){      /* 在 Filter 实例被 Web 容器从服务移除之前调用 */   }}

这边使用前文提到的 DisplayHeader.java 为例子:

//导入必需的 java 库import java.io.IOException;import java.io.PrintWriter;import java.util.Enumeration;import javax.servlet.ServletException;import javax.servlet.annotation.WebServlet;import javax.servlet.http.HttpServlet;import javax.servlet.http.HttpServletRequest;import javax.servlet.http.HttpServletResponse;@WebServlet("/DisplayHeader")//扩展 HttpServlet 类public class DisplayHeader extends HttpServlet {	// 处理 GET 方法请求的方法	public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException	{		// 设置响应内容类型		response.setContentType("text/html;charset=UTF-8");		PrintWriter out = response.getWriter();		String title = "HTTP Header 请求实例 - 51coolma教程实例";		String docType =			"<!DOCTYPE html> 
";			out.println(docType +			"<html>
" +			"<head><meta charset="utf-8"><title>" + title + "</title></head>
"+			"<body bgcolor="#f0f0f0">
" +			"<h1 align="center">" + title + "</h1>
" +			"<table width="100%" border="1" align="center">
" +			"<tr bgcolor="#949494">
" +			"<th>Header 名称</th><th>Header 值</th>
"+			"</tr>
");		Enumeration headerNames = request.getHeaderNames();		while(headerNames.hasMoreElements()) {			String paramName = (String)headerNames.nextElement();			out.print("<tr><td>" + paramName + "</td>
");			String paramValue = request.getHeader(paramName);			out.println("<td> " + paramValue + "</td></tr>
");		}		out.println("</table>
</body></html>");	}	// 处理 POST 方法请求的方法	public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {		doGet(request, response);	}}

Web.xml 中的 Servlet 过滤器映射(Servlet Filter Mapping)

定义过滤器,然后映射到一个 URL 或 Servlet,这与定义 Servlet,然后映射到一个 URL 模式方式大致相同。在部署描述符文件 web.xml 中为 filter 标签创建下面的条目:

<?xml version="1.0" encoding="UTF-8"?>  <web-app>  <filter>  <filter-name>LogFilter</filter-name>  <filter-class>com.51coolma.test.LogFilter</filter-class>  <init-param>    <param-name>Site</param-name>    <param-value>51coolma在线教程</param-value>  </init-param></filter><filter-mapping>  <filter-name>LogFilter</filter-name>  <url-pattern>/*</url-pattern></filter-mapping><servlet>    <!-- 类名 -->    <servlet-name>DisplayHeader</servlet-name>    <!-- 所在的包 -->    <servlet-class>com.51coolma.test.DisplayHeader</servlet-class>  </servlet>  <servlet-mapping>    <servlet-name>DisplayHeader</servlet-name>    <!-- 访问的网址 -->    <url-pattern>/TomcatTest/DisplayHeader</url-pattern>  </servlet-mapping>  </web-app>  

上述过滤器适用于所有的 Servlet,因为我们在配置中指定 /* 。如果您只想在少数的 Servlet 上应用过滤器,您可以指定一个特定的 Servlet 路径。

现在试着以常用的方式调用任何 Servlet,您将会看到在 Web 服务器中生成的日志。您也可以使用 Log4J 记录器来把上面的日志记录到一个单独的文件中。

使用多个过滤器

Web 应用程序可以根据特定的目的定义若干个不同的过滤器。假设您定义了两个过滤器 AuthenFilterLogFilter。您需要创建一个如下所述的不同的映射,其余的处理与上述所讲解的大致相同:

<filter>   <filter-name>LogFilter</filter-name>   <filter-class>com.51coolma.test.LogFilter</filter-class>   <init-param>	  <param-name>test-param</param-name>	  <param-value>Initialization Paramter</param-value>   </init-param></filter><filter>   <filter-name>AuthenFilter</filter-name>   <filter-class>com.51coolma.test.AuthenFilter</filter-class>   <init-param>	  <param-name>test-param</param-name>	  <param-value>Initialization Paramter</param-value>   </init-param></filter><filter-mapping>   <filter-name>LogFilter</filter-name>   <url-pattern>/*</url-pattern></filter-mapping><filter-mapping>   <filter-name>AuthenFilter</filter-name>   <url-pattern>/*</url-pattern></filter-mapping>

过滤器的应用顺序

web.xml 中的 filter-mapping 元素的顺序决定了 Web 容器应用过滤器到 Servlet 的顺序。若要反转过滤器的顺序,您只需要在 web.xml 文件中反转 filter-mapping 元素即可。

例如,上面的实例将先应用 LogFilter,然后再应用 AuthenFilter,但是下面的实例将颠倒这个顺序:

<filter-mapping>   <filter-name>AuthenFilter</filter-name>   <url-pattern>/*</url-pattern></filter-mapping><filter-mapping>   <filter-name>LogFilter</filter-name>   <url-pattern>/*</url-pattern></filter-mapping>

web.xml配置各节点说明

<filter>指定一个过滤器。
<filter-name>用于为过滤器指定一个名字,该元素的内容不能为空。
<filter-class>元素用于指定过滤器的完整的限定类名。
<init-param>元素用于为过滤器指定初始化参数,它的子元素<param-name>指定参数的名字,<param-value>指定参数的值。
在过滤器中,可以使用FilterConfig接口对象来访问初始化参数。
<filter-mapping>元素用于设置一个 Filter 所负责拦截的资源。一个Filter拦截的资源可通过两种方式来指定:Servlet 名称和资源访问的请求路径
<filter-name>子元素用于设置filter的注册名称。该值必须是在<filter>元素中声明过的过滤器的名字
<url-pattern>设置 filter 所拦截的请求路径(过滤器关联的URL样式)
<servlet-name>指定过滤器所拦截的Servlet名称。
<dispatcher>指定过滤器所拦截的资源被 Servlet 容器调用的方式,可以是REQUEST,INCLUDE,FORWARD和ERROR之一,默认REQUEST。用户可以设置多个<dispatcher>子元素用来指定 Filter 对资源的多种调用方式进行拦截。
<dispatcher>子元素可以设置的值及其意义
REQUEST:当用户直接访问页面时,Web容器将会调用过滤器。如果目标资源是通过RequestDispatcher的include()或forward()方法访问时,那么该过滤器就不会被调用。
INCLUDE:如果目标资源是通过RequestDispatcher的include()方法访问时,那么该过滤器将被调用。除此之外,该过滤器不会被调用。
FORWARD:如果目标资源是通过RequestDispatcher的forward()方法访问时,那么该过滤器将被调用,除此之外,该过滤器不会被调用。
ERROR:如果目标资源是通过声明式异常处理机制调用时,那么该过滤器将被调用。除此之外,过滤器不会被调用。

Servlet 异常处理

当一个 Servlet 抛出一个异常时,Web 容器在使用了 exception-type 元素的 web.xml 中搜索与抛出异常类型相匹配的配置。

您必须在 web.xml 中使用 error-page 元素来指定对特定异常 或 HTTP 状态码 作出相应的 Servlet 调用。

web.xml 配置

假设,有一个 ErrorHandler 的 Servlet 在任何已定义的异常或错误出现时被调用。以下将是在 web.xml 中创建的项。

<!-- servlet 定义 --><servlet>        <servlet-name>ErrorHandler</servlet-name>        <servlet-class>ErrorHandler</servlet-class></servlet><!-- servlet 映射 --><servlet-mapping>        <servlet-name>ErrorHandler</servlet-name>        <url-pattern>/ErrorHandler</url-pattern></servlet-mapping><!-- error-code 相关的错误页面 --><error-page>    <error-code>404</error-code>    <location>/ErrorHandler</location></error-page><error-page>    <error-code>403</error-code>    <location>/ErrorHandler</location></error-page><!-- exception-type 相关的错误页面 --><error-page>    <exception-type>          javax.servlet.ServletException    </exception-type >    <location>/ErrorHandler</location></error-page><error-page>    <exception-type>java.io.IOException</exception-type >    <location>/ErrorHandler</location></error-page>

如果您想对所有的异常有一个通用的错误处理程序,那么应该定义下面的 error-page,而不是为每个异常定义单独的 error-page 元素:

<error-page>    <exception-type>java.lang.Throwable</exception-type >    <location>/ErrorHandler</location></error-page>

以下是关于上面的 web.xml 异常处理要注意的点:

  • Servelt ErrorHandler 与其他的 Servelt 的定义方式一样,且在 web.xml 中进行配置。
  • 如果有错误状态代码出现,不管为 404(Not Found 未找到)或 403(Forbidden 禁止),则会调用 ErrorHandler 的 Servlet。
  • 如果 Web 应用程序抛出 ServletExceptionIOException,那么 Web 容器会调用 ErrorHandler 的 Servlet。
  • 您可以定义不同的错误处理程序来处理不同类型的错误或异常。上面的实例是非常通用的,希望您能通过实例理解基本的概念。

请求属性 - 错误/异常

以下是错误处理的 Servlet 可以访问的请求属性列表,用来分析错误/异常的性质。

序号属性 & 描述
1javax.servlet.error.status_code
该属性给出状态码,状态码可被存储,并在存储为 java.lang.Integer 数据类型后可被分析。
2javax.servlet.error.exception_type
该属性给出异常类型的信息,异常类型可被存储,并在存储为 java.lang.Class 数据类型后可被分析。
3javax.servlet.error.message
该属性给出确切错误消息的信息,信息可被存储,并在存储为 java.lang.String 数据类型后可被分析。
4javax.servlet.error.request_uri
该属性给出有关 URL 调用 Servlet 的信息,信息可被存储,并在存储为 java.lang.String 数据类型后可被分析。
5javax.servlet.error.exception
该属性给出异常产生的信息,信息可被存储,并在存储为 java.lang.Throwable 数据类型后可被分析。
6javax.servlet.error.servlet_name
该属性给出 Servlet 的名称,名称可被存储,并在存储为 java.lang.String 数据类型后可被分析。

Servlet 错误处理程序实例

以下是 Servlet 实例,将应对任何您所定义的错误或异常发生时的错误处理程序。

本实例让您对 Servlet 中的异常处理有基本的了解,您可以使用相同的概念编写更复杂的异常处理应用程序:

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*;import java.util.*;// 扩展 HttpServlet 类public class ErrorHandler extends HttpServlet {   // 处理 GET 方法请求的方法  public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 分析 Servlet 异常             Throwable throwable = (Throwable)      request.getAttribute("javax.servlet.error.exception");      Integer statusCode = (Integer)      request.getAttribute("javax.servlet.error.status_code");      String servletName = (String)      request.getAttribute("javax.servlet.error.servlet_name");      if (servletName == null){         servletName = "Unknown";      }      String requestUri = (String)      request.getAttribute("javax.servlet.error.request_uri");      if (requestUri == null){         requestUri = "Unknown";      }      // 设置响应内容类型      response.setContentType("text/html");       PrintWriter out = response.getWriter();     String title = "Error/Exception Information";      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +        "<html>
" +        "<head><title>" + title + "</title></head>
" +        "<body bgcolor="#f0f0f0">
");      if (throwable == null && statusCode == null){         out.println("<h2>Error information is missing</h2>");         out.println("Please return to the <a href="" +             response.encodeURL("http://localhost:8080/") +             "">Home Page</a>.");      }else if (statusCode != null){         out.println("The status code : " + statusCode);      }else{         out.println("<h2 class="tutheader">Error information</h2>");         out.println("Servlet Name : " + servletName +                              "</br></br>");         out.println("Exception Type : " +                              throwable.getClass( ).getName( ) +                              "</br></br>");         out.println("The request URI: " + requestUri +                              "<br><br>");         out.println("The exception message: " +                                  throwable.getMessage( ));      }      out.println("</body>");      out.println("</html>");  }  // 处理 POST 方法请求的方法  public void doPost(HttpServletRequest request,                     HttpServletResponse response)      throws ServletException, IOException {     doGet(request, response);  }}

以通常的方式编译 ErrorHandler.java,把您的类文件放入<Tomcat-installation-directory>/webapps/ROOT/WEB-INF/classes 中。

让我们在 web.xml 文件中添加如下配置来处理异常:

<servlet>        <servlet-name>ErrorHandler</servlet-name>        <servlet-class>ErrorHandler</servlet-class></servlet><!-- servlet mappings --><servlet-mapping>        <servlet-name>ErrorHandler</servlet-name>        <url-pattern>/ErrorHandler</url-pattern></servlet-mapping><error-page>    <error-code>404</error-code>    <location>/ErrorHandler</location></error-page><error-page>    <exception-type>java.lang.Throwable</exception-type >    <location>/ErrorHandler</location></error-page>

现在,尝试使用一个会产生异常的 Servlet,或者输入一个错误的 URL,这将触发 Web 容器调用 ErrorHandler 的 Servlet,并显示适当的消息。例如,如果您输入了一个错误的 URL,那么它将显示下面的结果:

The status code : 404

上面的代码在某些 Web 浏览器中可能无法正常工作。因此,请尽量尝试使用 Mozilla 和 Safari 浏览器,在这两种浏览器中它们应该能够正常工作。


Servlet Cookies 处理

Cookies 是存储在客户端计算机上的文本文件,并保留了各种跟踪信息。Java Servlet 显然支持 HTTP Cookies。

识别返回用户包括三个步骤:

  • 服务器脚本向浏览器发送一组 Cookies。例如:姓名、年龄或识别号码等。
  • 浏览器将这些信息存储在本地计算机上,以备将来使用。
  • 当下一次浏览器向 Web 服务器发送任何请求时,浏览器会把这些 Cookies 信息发送到服务器,服务器将使用这些信息来识别用户。

本章将向您讲解如何设置或重置 Cookies,如何访问它们,以及如何将它们删除。

Servlet Cookie 处理需要对中文进行编码与解码,方法如下:

String   str   =   java.net.URLEncoder.encode("中文");            //编码String   str   =   java.net.URLDecoder.decode("编码后的字符串");   // 解码

Cookie 剖析

Cookies 通常设置在 HTTP 头信息中(虽然 JavaScript 也可以直接在浏览器上设置一个 Cookie)。设置 Cookie 的 Servlet 会发送如下的头信息:

HTTP/1.1 200 OKDate: Fri, 04 Feb 2000 21:03:38 GMTServer: Apache/1.3.9 (UNIX) PHP/4.0b3Set-Cookie: name=xyz; expires=Friday, 04-Feb-07 22:03:38 GMT;                  path=/; domain=51coolma.cnConnection: closeContent-Type: text/html

正如您所看到的,Set-Cookie 头包含了一个名称值对、一个 GMT 日期、一个路径和一个域。名称和值会被 URL 编码。expires 字段是一个指令,告诉浏览器在给定的时间和日期之后"忘记"该 Cookie。

如果浏览器被配置为存储 Cookies,它将会保留此信息直到到期日期。如果用户的浏览器指向任何匹配该 Cookie 的路径和域的页面,它会重新发送 Cookie 到服务器。浏览器的头信息可能如下所示:

GET / HTTP/1.0Connection: Keep-AliveUser-Agent: Mozilla/4.6 (X11; I; Linux 2.2.6-15apmac ppc)Host: zink.demon.co.uk:1126Accept: image/gif, */*Accept-Encoding: gzipAccept-Language: enAccept-Charset: iso-8859-1,*,utf-8Cookie: name=xyz

Servlet 就能够通过请求方法 request.getCookies() 访问 Cookie,该方法将返回一个 Cookie 对象的数组。

Servlet Cookies 方法

以下是在 Servlet 中操作 Cookies 时可使用的有用的方法列表。

序号方法 & 描述
1public void setDomain(String pattern)
该方法设置 cookie 适用的域,例如 51coolma.cn。
2public String getDomain()
该方法获取 cookie 适用的域,例如 51coolma.cn。
3public void setMaxAge(int expiry)
该方法设置 cookie 过期的时间(以秒为单位)。如果不这样设置,cookie 只会在当前 session 会话中持续有效。
4public int getMaxAge()
该方法返回 cookie 的最大生存周期(以秒为单位),默认情况下,-1 表示 cookie 将持续下去,直到浏览器关闭。
5public String getName()
该方法返回 cookie 的名称。名称在创建后不能改变。
6public void setValue(String newValue)
该方法设置与 cookie 关联的值。
7public String getValue()
该方法获取与 cookie 关联的值。
8public void setPath(String uri)
该方法设置 cookie 适用的路径。如果您不指定路径,与当前页面相同目录下的(包括子目录下的)所有 URL 都会返回 cookie。
9public String getPath()
该方法获取 cookie 适用的路径。
10public void setSecure(boolean flag)
该方法设置布尔值,表示 cookie 是否应该只在加密的(即 SSL)连接上发送。
11public void setComment(String purpose)
该方法规定了描述 cookie 目的的注释。该注释在浏览器向用户呈现 cookie 时非常有用。
12public String getComment()
该方法返回了描述 cookie 目的的注释,如果 cookie 没有注释则返回 null。

通过 Servlet 设置 Cookies

通过 Servlet 设置 Cookies 包括三个步骤:

(1) 创建一个 Cookie 对象:您可以调用带有 cookie 名称和 cookie 值的 Cookie 构造函数,cookie 名称和 cookie 值都是字符串。

Cookie cookie = new Cookie("key","value");

请记住,无论是名字还是值,都不应该包含空格或以下任何字符:

[ ] ( ) = , " / ? @ : ;

(2) 设置最大生存周期:您可以使用 setMaxAge 方法来指定 cookie 能够保持有效的时间(以秒为单位)。下面将设置一个最长有效期为 24 小时的 cookie。

cookie.setMaxAge(60*60*24); 

(3) 发送 Cookie 到 HTTP 响应头:您可以使用 response.addCookie 来添加 HTTP 响应头中的 Cookies,如下所示:

response.addCookie(cookie);

实例

让我们修改我们的 表单数据实例,为名字和姓氏设置 Cookies。

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*; // 扩展 HttpServlet 类public class HelloForm extends HttpServlet {   public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 为名字和姓氏创建 Cookies            Cookie firstName = new Cookie("first_name",                      request.getParameter("first_name"));      Cookie lastName = new Cookie("last_name",                      request.getParameter("last_name"));      // 为两个 Cookies 设置过期日期为 24 小时后      firstName.setMaxAge(60*60*24);       lastName.setMaxAge(60*60*24);       // 在响应头中添加两个 Cookies      response.addCookie( firstName );      response.addCookie( lastName );      // 设置响应内容类型      response.setContentType("text/html");       PrintWriter out = response.getWriter();      String title = "设置 Cookies 实例";      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +                "<html>
" +                "<head><title>" + title + "</title></head>
" +                "<body bgcolor="#f0f0f0">
" +                "<h1 align="center">" + title + "</h1>
" +                "<ul>
" +                "  <li><b>名字</b>:"                + request.getParameter("first_name") + "
" +                "  <li><b>姓氏</b>:"                + request.getParameter("last_name") + "
" +                "</ul>
" +                "</body></html>");  }}

编译上面的 Servlet HelloForm,并在 web.xml 文件中创建适当的条目,最后尝试下面的 HTML 页面来调用 Servlet。

 <html><body><form action="HelloForm" method="GET">名字:<input type="text" name="first_name"><br />姓氏:<input type="text" name="last_name" /><input type="submit" value="提交" /></form></body></html>

保存上面的 HTML 内容到文件 hello.htm 中,并把它放在 <Tomcat-installation-directory>/webapps/ROOT 目录中。当您访问 http://localhost:8080/Hello.htm 时,上面表单的实际输出如下所示:


尝试输入名字和姓氏,然后点击"提交"按钮,名字和姓氏将显示在屏幕上,同时会设置 firstName 和 lastName 这两个 Cookies,当下次您按下提交按钮时,会将这两个 Cookies 传回到服务器。

下一节会讲解如何在 Web 应用程序中访问这些 Cookies。

通过 Servlet 读取 Cookies

要读取 Cookies,您需要通过调用 HttpServletRequestgetCookies( ) 方法创建一个 javax.servlet.http.Cookie 对象的数组。然后循环遍历数组,并使用 getName() 和 getValue() 方法来访问每个 cookie 和关联的值。

实例

让我们读取上面的实例中设置的 Cookies

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*; // 扩展 HttpServlet 类public class ReadCookies extends HttpServlet {   public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      Cookie cookie = null;    Cookie[] cookies = null;      // 获取与该域相关的 Cookies 的数组      cookies = request.getCookies();          // 设置响应内容类型      response.setContentType("text/html");       PrintWriter out = response.getWriter();      String title = "Reading Cookies Example";      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +                "<html>
" +                "<head><title>" + title + "</title></head>
" +                "<body bgcolor="#f0f0f0">
" );      if( cookies != null ){         out.println("<h2>查找 Cookies 名称和值</h2>");         for (int i = 0; i < cookies.length; i++){             cookie = cookies[i];             out.print("名称:" + cookie.getName( ) + ",");             out.print("值:" + cookie.getValue( )+" <br/>");         }      }else{          out.println(            "<h2 class="tutheader">未找到 Cookies</h2>");      }      out.println("</body>");      out.println("</html>");   }}

编译上面的 Servlet ReadCookies,并在 web.xml 文件中创建适当的条目。如果您已经设置了 first_name cookie 为 "John",last_name cookie 为 "Player" ,尝试运行 http://localhost:8080/ReadCookies,将显示如下结果:

查找 Cookies 名称和值

名称:first_name,值:John
名称:last_name,值:Player

通过 Servlet 删除 Cookies

删除 Cookies 是非常简单的。如果您想删除一个 cookie,那么您只需要按照以下三个步骤进行:

  • 读取一个现有的 cookie,并把它存储在 Cookie 对象中。
  • 使用 setMaxAge() 方法设置 cookie 的年龄为零,来删除现有的 cookie。
  • 把这个 cookie 添加到响应头。

实例

下面的例子将删除现有的名为 "first_name" 的 cookie,当您下次运行 ReadCookies 的 Servlet 时,它会返回 first_name 为空值。

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*; // 扩展 HttpServlet 类public class DeleteCookies extends HttpServlet {   public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      Cookie cookie = null;    Cookie[] cookies = null;      // 获取与该域相关的 Cookies 的数组      cookies = request.getCookies();          // 设置响应内容类型      response.setContentType("text/html");       PrintWriter out = response.getWriter();      String title = "Delete Cookies Example";      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +                "<html>
" +                "<head><title>" + title + "</title></head>
" +                "<body bgcolor="#f0f0f0">
" );       if( cookies != null ){         out.println("<h2>Cookies 名称和值</h2>");         for (int i = 0; i < cookies.length; i++){             cookie = cookies[i];             if((cookie.getName( )).compareTo("first_name") == 0 ){                  cookie.setMaxAge(0);                  response.addCookie(cookie);                  out.print("已删除的 cookie:" +                                cookie.getName( ) + "<br/>");            }            out.print("名称:" + cookie.getName( ) + ",");            out.print("值:" + cookie.getValue( )+" <br/>");         }      }else{          out.println(            "<h2 class="tutheader">No cookies founds</h2>");      }      out.println("</body>");      out.println("</html>");   }}

编译上面的 Servlet DeleteCookies,并在 web.xml 文件中创建适当的条目。现在运行 http://localhost:8080/DeleteCookies,将显示如下结果:

Cookies 名称和值

已删除的 cookie:first_name
名称:first_name,值:John
名称:last_name,值:Player

现在尝试运行 http://localhost:8080/ReadCookies,它将只显示一个 cookie,如下所示:

查找 Cookies 名称和值

名称:last_name,值:Player

您可以手动在 Internet Explorer 中删除 Cookies。在"工具"菜单,选择"Internet 选项"。如果要删除所有的 Cookies,请按"删除 Cookies"。

Servlet Session 跟踪

HTTP 是一种"无状态"协议,这意味着每次客户端检索网页时,客户端打开一个单独的连接到 Web 服务器,服务器会自动不保留之前客户端请求的任何记录。

但是仍然有以下三种方式来维持 Web 客户端和 Web 服务器之间的 session 会话:

Cookies

一个 Web 服务器可以分配一个唯一的 session 会话 ID 作为每个 Web 客户端的 cookie,对于客户端的后续请求可以使用接收到的 cookie 来识别。

这可能不是一个有效的方法,因为很多浏览器不支持 cookie,所以我们建议不要使用这种方式来维持 session 会话。

隐藏的表单字段

一个 Web 服务器可以发送一个隐藏的 HTML 表单字段,以及一个唯一的 session 会话 ID,如下所示:

<input type="hidden" name="sessionid" value="12345">

该条目意味着,当表单被提交时,指定的名称和值会被自动包含在 GET 或 POST 数据中。每次当 Web 浏览器发送回请求时,session_id 值可以用于保持不同的 Web 浏览器的跟踪。

这可能是一种保持 session 会话跟踪的有效方式,但是点击常规的超文本链接(<A HREF...>)不会导致表单提交,因此隐藏的表单字段也不支持常规的 session 会话跟踪。

URL 重写

您可以在每个 URL 末尾追加一些额外的数据来标识 session 会话,服务器会把该 session 会话标识符与已存储的有关 session 会话的数据相关联。

例如,http://51coolma.cn/file.htm;sessionid=12345,session 会话标识符被附加为 sessionid=12345,标识符可被 Web 服务器访问以识别客户端。

URL 重写是一种更好的维持 session 会话的方式,它在浏览器不支持 cookie 时能够很好地工作,但是它的缺点是会动态生成每个 URL 来为页面分配一个 session 会话 ID,即使是在很简单的静态 HTML 页面中也会如此。

HttpSession 对象

除了上述的三种方式,Servlet 还提供了 HttpSession 接口,该接口提供了一种跨多个页面请求或访问网站时识别用户以及存储有关用户信息的方式。

Servlet 容器使用这个接口来创建一个 HTTP 客户端和 HTTP 服务器之间的 session 会话。会话持续一个指定的时间段,跨多个连接或页面请求。

您会通过调用 HttpServletRequest 的公共方法 getSession() 来获取 HttpSession 对象,如下所示:

HttpSession session = request.getSession();

你需要在向客户端发送任何文档内容之前调用 request.getSession()。下面总结了 HttpSession 对象中可用的几个重要的方法:

序号方法 & 描述
1public Object getAttribute(String name)
该方法返回在该 session 会话中具有指定名称的对象,如果没有指定名称的对象,则返回 null。
2public Enumeration getAttributeNames()
该方法返回 String 对象的枚举,String 对象包含所有绑定到该 session 会话的对象的名称。
3public long getCreationTime()
该方法返回该 session 会话被创建的时间,自格林尼治标准时间 1970 年 1 月 1 日午夜算起,以毫秒为单位。
4public String getId()
该方法返回一个包含分配给该 session 会话的唯一标识符的字符串。
5public long getLastAccessedTime()
该方法返回客户端最后一次发送与该 session 会话相关的请求的时间自格林尼治标准时间 1970 年 1 月 1 日午夜算起,以毫秒为单位。
6public int getMaxInactiveInterval()
该方法返回 Servlet 容器在客户端访问时保持 session 会话打开的最大时间间隔,以秒为单位。
7public void invalidate()
该方法指示该 session 会话无效,并解除绑定到它上面的任何对象。
8public boolean isNew(
如果客户端还不知道该 session 会话,或者如果客户选择不参入该 session 会话,则该方法返回 true。
9public void removeAttribute(String name)
该方法将从该 session 会话移除指定名称的对象。
10public void setAttribute(String name, Object value)
该方法使用指定的名称绑定一个对象到该 session 会话。
11public void setMaxInactiveInterval(int interval)
该方法在 Servlet 容器指示该 session 会话无效之前,指定客户端请求之间的时间,以秒为单位。

Session 跟踪实例

本实例说明了如何使用 HttpSession 对象获取 session 会话创建时间和最后访问时间。如果不存在 session 会话,我们将通过请求创建一个新的 session 会话。

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*;import java.util.*; // 扩展 HttpServlet 类public class SessionTrack extends HttpServlet {   public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 如果不存在 session 会话,则创建一个 session 对象      HttpSession session = request.getSession(true);      // 获取 session 创建时间      Date createTime = new Date(session.getCreationTime());      // 获取该网页的最后一次访问时间      Date lastAccessTime =                         new Date(session.getLastAccessedTime());      String title = "欢迎回到我的网站";      Integer visitCount = new Integer(0);      String visitCountKey = new String("visitCount");      String userIDKey = new String("userID");      String userID = new String("ABCD");      // 检查网页上是否有新的访问者      if (session.isNew()){         title = "欢迎来到我的网站";         session.setAttribute(userIDKey, userID);      } else {         visitCount = (Integer)session.getAttribute(visitCountKey);         visitCount = visitCount + 1;         userID = (String)session.getAttribute(userIDKey);      }      session.setAttribute(visitCountKey,  visitCount);      // 设置响应内容类型      response.setContentType("text/html");      PrintWriter out = response.getWriter();      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +                "<html>
" +                "<head><title>" + title + "</title></head>
" +                "<body bgcolor="#f0f0f0">
" +                "<h1 align="center">" + title + "</h1>
" +                 "<h2 align="center">Session 信息</h2>
" +                "<table border="1" align="center">
" +                "<tr bgcolor="#949494">
" +                "  <th>Session 信息</th><th>值</th></tr>
" +                "<tr>
" +                "  <td>id</td>
" +                "  <td>" + session.getId() + "</td></tr>
" +                "<tr>
" +                "  <td>Creation Time</td>
" +                "  <td>" + createTime +                 "  </td></tr>
" +                "<tr>
" +                "  <td>Time of Last Access</td>
" +                "  <td>" + lastAccessTime +                 "  </td></tr>
" +                "<tr>
" +                "  <td>User ID</td>
" +                "  <td>" + userID +                 "  </td></tr>
" +                "<tr>
" +                "  <td>Number of visits</td>
" +                "  <td>" + visitCount + "</td></tr>
" +                "</table>
" +                "</body></html>");  }}

编译上面的 Servlet SessionTrack,并在 web.xml 文件中创建适当的条目。在浏览器地址栏输入 http://localhost:8080/SessionTrack,当您第一次运行时将显示如下结果:

欢迎来到我的网站

Session 信息

Session 信息
id 0AE3EC93FF44E3C525B4351B77ABB2D5
Creation Time Tue Jun 08 17:26:40 GMT+04:00 2014
Time of Last Access Tue Jun 08 17:26:40 GMT+04:00 2014
User ID ABCD
Number of visits 0

再次尝试运行相同的 Servlet,它将显示如下结果:

欢迎回到我的网站

Session 信息

Session 信息
id 0AE3EC93FF44E3C525B4351B77ABB2D5
Creation Time Tue Jun 08 17:26:40 GMT+04:00 2014
Time of Last Access Tue Jun 08 17:26:40 GMT+04:00 2014
User ID ABCD
Number of visits 1

删除 Session 会话数据

当您完成了一个用户的 session 会话数据,您有以下几种选择:

  • 移除一个特定的属性:您可以调用 public void removeAttribute(String name) 方法来删除与特定的键相关联的值。 to delete the value associated with a particular key.
  • 删除整个 session 会话:您可以调用 public void invalidate() 方法来丢弃整个 session 会话。
  • 设置 session 会话过期时间:您可以调用 public void setMaxInactiveInterval(int interval) 方法来单独设置 session 会话超时。
  • 注销用户:如果使用的是支持 servlet 2.4 的服务器,您可以调用 logout 来注销 Web 服务器的客户端,并把属于所有用户的所有 session 会话设置为无效。
  • web.xml 配置:如果您使用的是 Tomcat,除了上述方法,您还可以在 web.xml 文件中配置 session 会话超时,如下所示:
  <session-config>    <session-timeout>15</session-timeout>  </session-config>

上面实例中的超时时间是以分钟为单位,将覆盖 Tomcat 中默认的 30 分钟超时时间。

在一个 Servlet 中的 getMaxInactiveInterval() 方法会返回 session 会话的超时时间,以秒为单位。所以,如果在 web.xml 中配置 session 会话超时时间为 15 分钟,那么 getMaxInactiveInterval() 会返回 900。

ServletContext是一个全局的储存信息的空间,服务器开始就存在,服务器关闭才释放。为了方便大家理解,我们将ServletContext和Cookie、Session做一个简单对比,如下图:


1


我们可以把ServletContext当成一个公用的空间,可以被所有的客户访问,如上图,A、B、C三个客户端都可以访问。

WEB容器在启动时,它会为每个Web应用程序都创建一个对应的ServletContext,它代表当前Web应用,并且它被所有客户端共享。


由于一个WEB应用中的所有Servlet共享同一个ServletContext对象,因此Servlet对象之间可以通过ServletContext对象来实现通讯。ServletContext对象通常也被称之为context域对象。公共聊天室就会用到它。

当web应用关闭、Tomcat关闭或者Web应用reload的时候,ServletContext对象会被销毁


ServletContext使用方法


1、ServletContext对象如何得到

this.getServletContext(); this.getServletConfig().getServletContext();

2、你可以把它想象成一张表,这个和Session非常相似:每一行就是一个属性,如下:

名字(String)值(Object)


添加属性:setAttribute(String name, Object obj);

得到值:getAttribute(String name),这个方法返回Object

删除属性:removeAttribute(String name)


3、生命周期 

ServletContext中的属性的生命周期从创建开始,到服务器关闭结束。


一个快速入门的案例: 

我们创建Servlet1和Servlet2,分别用于在ServletContext中创建和读取属性: 


Servlet1的doGet方法为:

public void doGet(HttpServletRequest request, HttpServletResponse response)        throws ServletException, IOException {    response.setContentType("text/html;charset=utf-8");    PrintWriter out = response.getWriter();    // 获取ServletContext对象的引用    // 第一种方法    ServletContext servletContext = this.getServletContext();    // 第二种方法    // ServletContext servletContext2 = this.getServletConfig().getServletContext();    servletContext.setAttribute("name", "小明");    out.println("将 name=小明  写入了ServletContext");}

Servlet2的doGet方法为:

public void doGet(HttpServletRequest request, HttpServletResponse response)        throws ServletException, IOException {    response.setContentType("text/html;charset=utf-8");    PrintWriter out = response.getWriter();    // 取出ServletContext的某个属性    //1.首先获取到ServletContext    ServletContext servletContext = this.getServletContext();    //2.取出属性    String name = (String)servletContext.getAttribute("name");    out.println("name="+name);}

以此访问Servlet1和Servlet2,我们可以分别看到输出如下:


Servletcontext 对象


Servletcontext 对象


粗看之下,这个运行结果和Session,Cookie的应用似乎没什么区别,但事实上则完全不一样的。只要不关闭Tomcat或者reload该应用,当我们关闭当前的浏览器,或者是换一个浏览器,比如从360浏览器换到了IE浏览器再次访问Servlet2,我们依然可以看到这个结果!这就是和和Session,Cookie最大的不同了。之所以会造成这种不同,是因为ServletContext存在于服务器内存中的一个公共空间,它可以供所有的用户客户端访问。


ServletContext应用


1、多个Servlet可以通过ServletContext对象来实现数据间的共享

类似于Session,通过ServletContext对象我们也可以实现数据共享,但值得注意的是,Session是只能在一个客户端中共享数据,而ServletContext中的数据是在所有客户端中都可以实现数据共享的。


2、实现Servlet的请求转发

之前我们学过的请求转发是通过request对象的: 

request.getRequestDispatcher("/url").forward(request, response);

这里要说明的是,ServletContext也可以实现请求转发: 

this.getServletContext().getRequestDispatcher("/url").forward(request, response); 

这两个转发效果是一样的。


3、获取Web应用的初始化参数

我们可以用<init-param>标签为servlet配置初始化参数,然后使用ServletConfig对象获取这些参数,假如有如下的MyServlet,它的配置为:

<servlet>      <servlet-name>MyServlet</servlet-name>      <servlet-class>com.gavin.servlet.MyServlet</servlet-class>      <init-param>          <param-name>encoding</param-name>          <param-value>utf-8</param-value>      </init-param>  </servlet>  

可以看到它配置了一个初始化参数:encoding=utf-8,那么我们在MyServlet的源代码中需要这样去得到这个参数:

String encoding = this.getServletConfig().getInitParameter("encoding");

注意,上述的参数配置方法只针对一个特定的Servlet有效,我们可以通过ServletContext来获取全局的、整个Web应用的初始化参数,全局的初始化参数是这样配置在web.xml文件中的:

<!-- 如果希望所有的Servlet都可以使用该配置,则必须这么做 --><context-param>    <param-name>name</param-name>    <param-value>gavin</param-value></context-param>

然后我们可以在任意一个Servlet中使用ServletContext获取这个参数:

String name = this.getServletContext().getInitParameter("name");


4、利用ServletContext对象读取资源文件(比如properties文件) 

读取资源文件要根据资源文件所在的位置来决定,一般分为以下两种情况:


4.1:文件在WebRoot文件夹下,即Web应用的根目录。这时候我们可以使用ServletContext来读取该资源文件。

假设我们Web根目录下有一个配置数据库信息的dbinfo.properties文件,里面配置了name和password属性,这时候可以通过ServletContext去读取这个文件:

// 这种方法的默认读取路径就是Web应用的根目录InputStream stream = this.getServletContext().getResourceAsStream("dbinfo.properties");// 创建属性对象Properties properties = new Properties();properties.load(stream);String name = properties.getProperty("name");String password = properties.getProperty("password");out.println("name="+name+";password="+password);

4.2:如果这个文件放在了src目录下,这时就不能用ServletContext来读取了,必须要使用类加载器去读取。

// 类加载器的默认读取路径是src根目录InputStream stream = MyServlet.class.getClassLoader().getResourceAsStream("dbinfo.properties")

如果这个文件此时还没有直接在src目录下,而是在src目录下的某个包下,比如在com.gavin包下,此时类加载器要加上包的路径,如下:

InputStream stream = MyServlet.class.getClassLoader().getResourceAsStream("com/gavin/dbinfo.properties")

补充一点,ServletContext只有在读取的文件在web应用的根目录下时,才能获取文件的全路径。比如我们在WebRoot文件夹下有一个images文件夹,images文件夹下有一个Servlet.jpg图片,为了得到这个图片的全路径,如下:

// 如何读取到一个文件的全路径,这里会得到在Tomcat的全路径String path = this.getServletContext().getRealPath("/images/Servlet.jpg");

在网站开发中,有很多功能要使用ServletContext,比如 

1. 网站计数器 

2. 网站的在线用户显示 

3. 简单的聊天系统

总之,如果是涉及到不同用户共享数据,而这些数据量不大,同时又不希望写入数据库中,我们就可以考虑使用ServletContext实现。


ServletContext使用建议

因为存在ServletContext中的数据在服务器中会长时间,这样就会占用很多内存,因此在使用ServletContext时,建议不要往里面添加过大的数据!


相关阅读:

Servlet教程

Servlet API官方文档


Servlet 数据库访问

本教程假定您已经了解了 JDBC 应用程序的工作方式。在您开始学习 Servlet 数据库访问之前,请确保您已经有适当的 JDBC 环境设置和数据库。

从基本概念下手,让我们来创建一个简单的表,并在表中创建几条记录。

创建数据库表

在测试数据库 TEST 中创建 Employees 表,请按以下步骤进行:

步骤 1:

打开命令行提示符(Command Prompt),并更改进入到安装目录,如下所示:

C:>C:>cd Program FilesMySQLinC:Program FilesMySQLin>

步骤 2:

登录到数据库,如下所示:

C:Program FilesMySQLin>mysql -u root -pEnter password: ********mysql>

步骤 3:

在测试数据库 TEST 中创建 Employee 表,如下所示:

mysql> use TEST;mysql> create table Employees    (     id int not null,     age int not null,     first varchar (255),     last varchar (255)    );Query OK, 0 rows affected (0.08 sec)mysql>

创建数据记录

最后,在 Employee 表中创建几条记录,如下所示:

mysql> INSERT INTO Employees VALUES (100, 18, 'Zara', 'Ali');Query OK, 1 row affected (0.05 sec) mysql> INSERT INTO Employees VALUES (101, 25, 'Mahnaz', 'Fatma');Query OK, 1 row affected (0.00 sec) mysql> INSERT INTO Employees VALUES (102, 30, 'Zaid', 'Khan');Query OK, 1 row affected (0.00 sec) mysql> INSERT INTO Employees VALUES (103, 28, 'Sumit', 'Mittal');Query OK, 1 row affected (0.00 sec) mysql>

访问数据库

下面的实例演示了如何使用 Servlet 访问 TEST 数据库。

// 加载必需的库import java.io.*;import java.util.*;import javax.servlet.*;import javax.servlet.http.*;import java.sql.*; public class DatabaseAccess extends HttpServlet{      public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // JDBC 驱动器名称和数据库的 URL      static final String JDBC_DRIVER="com.mysql.jdbc.Driver";        static final String DB_URL="jdbc:mysql://localhost/TEST";      //  数据库的用户名与密码,需要根据自己的设置      static final String USER = "root";      static final String PASS = "password";      // 设置响应内容类型      response.setContentType("text/html");      PrintWriter out = response.getWriter();      String title = "数据库结果";      String docType =        "<!doctype html public "-//w3c//dtd html 4.0 " +          "transitional//en">
";         out.println(docType +         "<html>
" +         "<head><title>" + title + "</title></head>
" +         "<body bgcolor="#f0f0f0">
" +         "<h1 align="center">" + title + "</h1>
");      try{         // 注册 JDBC 驱动器         Class.forName("com.mysql.jdbc.Driver");         // 打开一个连接         Connection conn = DriverManager.getConnection(DB_URL,USER,PASS);         // 执行 SQL 查询         Statement stmt = conn.createStatement();         String sql;         sql = "SELECT id, first, last, age FROM Employees";         ResultSet rs = stmt.executeQuery(sql);         // 展开结果集数据库         while(rs.next()){            // 通过字段检索            int id  = rs.getInt("id");            int age = rs.getInt("age");            String first = rs.getString("first");            String last = rs.getString("last");            // 输出数据            out.println("ID: " + id + "<br>");            out.println(", Age: " + age + "<br>");            out.println(", First: " + first + "<br>");            out.println(", Last: " + last + "<br>");         }         out.println("</body></html>");         // 完成后关闭         rs.close();         stmt.close();         conn.close();      }catch(SQLException se){         // 处理 JDBC 错误         se.printStackTrace();      }catch(Exception e){         // 处理 Class.forName 错误         e.printStackTrace();      }finally{         // 最后是用于关闭资源的块         try{            if(stmt!=null)               stmt.close();         }catch(SQLException se2){         }         try{            if(conn!=null)            conn.close();         }catch(SQLException se){            se.printStackTrace();         }      }   }} 

现在让我们来编译上面的 Servlet,并在 web.xml 文件中创建以下条目:

.... <servlet>     <servlet-name>DatabaseAccess</servlet-name>     <servlet-class>DatabaseAccess</servlet-class> </servlet>  <servlet-mapping>     <servlet-name>DatabaseAccess</servlet-name>     <url-pattern>/DatabaseAccess</url-pattern> </servlet-mapping>....

现在调用这个 Servlet,输入链接:http://localhost:8080/DatabaseAccess,将显示以下响应结果:

数据库结果ID: 100, Age: 18, First: Zara, Last: AliID: 101, Age: 25, First: Mahnaz, Last: FatmaID: 102, Age: 30, First: Zaid, Last: KhanID: 103, Age: 28, First: Sumit, Last: Mittal

Servlet 处理日期

使用 Servlet 的最重要的优势之一是,可以使用核心 Java 中的大多数可用的方法。本章将讲解 Java 提供的 java.util 包中的 Date 类,这个类封装了当前的日期和时间。

Date 类支持两个构造函数。第一个构造函数初始化当前日期和时间的对象。

Date( )

下面的构造函数接受一个参数,该参数等于 1970 年 1 月 1 日午夜以来经过的毫秒数。

Date(long millisec)

一旦您有一个可用的 Date 对象,您可以调用下列任意支持的方法来使用日期:

序号方法 & 描述
1boolean after(Date date)
如果调用的 Date 对象中包含的日期在 date 指定的日期之后,则返回 true,否则返回 false。
2boolean before(Date date)
如果调用的 Date 对象中包含的日期在 date 指定的日期之前,则返回 true,否则返回 false。
3Object clone( )
重复调用 Date 对象。
4int compareTo(Date date)
把调用对象的值与 date 的值进行比较。如果两个值是相等的,则返回 0。如果调用对象在 date 之前,则返回一个负值。如果调用对象在 date 之后,则返回一个正值。
5int compareTo(Object obj)
如果 obj 是 Date 类,则操作等同于 compareTo(Date)。否则,它会抛出一个 ClassCastException。
6boolean equals(Object date)
如果调用的 Date 对象中包含的时间和日期与 date 指定的相同,则返回 true,否则返回 false。
7long getTime( )
返回 1970 年 1 月 1 日以来经过的毫秒数。
8int hashCode( )
为调用对象返回哈希代码。
9void setTime(long time)
设置 time 指定的时间和日期,这表示从 1970 年 1 月 1 日午夜以来经过的时间(以毫秒为单位)。
10String toString( )
转换调用的 Date 对象为一个字符串,并返回结果。

获取当前的日期和时间

在 Java Servlet 中获取当前的日期和时间是非常容易的。您可以使用一个简单的 Date 对象的 toString() 方法来输出当前的日期和时间,如下所示:

// 导入必需的 java 库import java.io.*;import java.util.Date;import javax.servlet.*;import javax.servlet.http.*; // 扩展 HttpServlet 类public class CurrentDate extends HttpServlet {   public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 设置响应内容类型      response.setContentType("text/html");       PrintWriter out = response.getWriter();      String title = "显示当前的日期和时间";      Date date = new Date();      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +        "<html>
" +        "<head><title>" + title + "</title></head>
" +        "<body bgcolor="#f0f0f0">
" +        "<h1 align="center">" + title + "</h1>
" +        "<h2 align="center">" + date.toString() + "</h2>
" +        "</body></html>");  }}

现在,让我们来编译上面的 Servlet,并在 web.xml 文件中创建适当的条目,然后通过访问 http://localhost:8080/CurrentDate 来调用该 Servlet。这将会产生如下的结果:

显示当前的日期和时间

Mon Jun 21 21:46:49 GMT+04:00 2010

尝试刷新 URL http://localhost:8080/CurrentDate,每隔几秒刷新一次您都会发现显示时间的差异。

日期比较

正如上面所提到的,您可以在 Servlet 中使用所有可用的 Java 方法。如果您需要比较两个日期,以下是方法:

  • 您可以使用 getTime() 来获取两个对象自 1970 年 1 月 1 日午夜以来经过的时间(以毫秒为单位),然后对这两个值进行比较。
  • 您可以使用方法 before( )、after( ) 和 equals( )。由于一个月里 12 号在 18 号之前,例如,new Date(99, 2, 12).before(new Date (99, 2, 18)) 返回 true。
  • 您可以使用 compareTo( ) 方法,该方法由 Comparable 接口定义,由 Date 实现。

使用 SimpleDateFormat 格式化日期

SimpleDateFormat 是一个以语言环境敏感的方式来格式化和解析日期的具体类。 SimpleDateFormat 允许您选择任何用户定义的日期时间格式化的模式。

让我们修改上面的实例,如下所示:

// 导入必需的 java 库import java.io.*;import java.text.*;import java.util.Date;import javax.servlet.*;import javax.servlet.http.*; // 扩展 HttpServlet 类public class CurrentDate extends HttpServlet {   public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 设置响应内容类型      response.setContentType("text/html");       PrintWriter out = response.getWriter();      String title = "显示当前的日期和时间";      Date dNow = new Date( );      SimpleDateFormat ft =       new SimpleDateFormat ("E yyyy.MM.dd 'at' hh:mm:ss a zzz");      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +        "<html>
" +        "<head><title>" + title + "</title></head>
" +        "<body bgcolor="#f0f0f0">
" +        "<h1 align="center">" + title + "</h1>
" +        "<h2 align="center">" + ft.format(dNow) + "</h2>
" +        "</body></html>");  }}

再次编译上面的 Servlet,然后通过访问 http://localhost:8080/CurrentDate 来调用该 Servlet。这将会产生如下的结果:

显示当前的日期和时间

Mon 2010.06.21 at 10:06:44 PM GMT+04:00


简单的日期格式的格式代码

使用事件模式字符串来指定时间格式。在这种模式下,所有的 ASCII 字母被保留为模式字母,这些字母定义如下:

字符描述实例
GEra 指示器AD
y四位数表示的年2001
M一年中的月July 或 07
d一月中的第几天10
h带有 A.M./P.M. 的小时(1~12)12
H一天中的第几小时(0~23)22
m一小时中的第几分30
s一分中的第几秒55
S毫秒234
E一周中的星期几Tuesday
D一年中的第几天360
F所在的周是这个月的第几周2 (second Wed. in July)
w一年中的第几周40
W一月中的第几周1
aA.M./P.M. 标记PM
k一天中的第几小时(1~24)24
K带有 A.M./P.M. 的小时(0~11)10
z时区Eastern Standard Time
'Escape for textDelimiter
"单引号`

如需查看可用的处理日期方法的完整列表,您可以参考标准的 Java 文档。

Servlet 网页重定向

当文档移动到新的位置,我们需要向客户端发送这个新位置时,我们需要用到网页重定向。当然,也可能是为了负载均衡,或者只是为了简单的随机,这些情况都有可能用到网页重定向。

重定向请求到另一个网页的最简单的方式是使用 response 对象的 sendRedirect() 方法。下面是该方法的定义: 将请求重定向到另一页的最简单的方法是,用方法的sendRedirect()的响应对象。以下是这种方法的定义:

public void HttpServletResponse.sendRedirect(String location)throws IOException 

该方法把响应连同状态码和新的网页位置发送回浏览器。您也可以通过把 setStatus() 和 setHeader() 方法一起使用来达到同样的效果:

....String site = "http://www.newpage.com" ;response.setStatus(response.SC_MOVED_TEMPORARILY);response.setHeader("Location", site); ....

实例

本实例显示了 Servlet 如何进行页面重定向到另一个位置:

import java.io.*;import java.sql.Date;import java.util.*;import javax.servlet.*;import javax.servlet.http.*;public class PageRedirect extends HttpServlet{      public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 设置响应内容类型      response.setContentType("text/html");      // 要重定向的新位置      String site = new String("http://www.51coolma.cn");      response.setStatus(response.SC_MOVED_TEMPORARILY);      response.setHeader("Location", site);        }} 

现在让我们来编译上面的 Servlet,并在 web.xml 文件中创建以下条目:

.... <servlet>     <servlet-name>PageRedirect</servlet-name>     <servlet-class>PageRedirect</servlet-class> </servlet> <servlet-mapping>     <servlet-name>PageRedirect</servlet-name>     <url-pattern>/PageRedirect</url-pattern> </servlet-mapping>....

现在通过访问 URL http://localhost:8080/PageRedirect 来调用这个 Servlet。这将把您转到给定的 URL http://www.51coolma.cn。

Servlet 点击计数器

网页点击计数器

很多时候,您可能有兴趣知道网站的某个特定页面上的总点击量。使用 Servlet 来计算这些点击量是非常简单的,因为一个 Servlet 的生命周期是由它运行所在的容器控制的。

以下是实现一个简单的基于 Servlet 生命周期的网页点击计数器需要采取的步骤:

  • 在 init() 方法中初始化一个全局变量。
  • 每次调用 doGet() 或 doPost() 方法时,都增加全局变量。
  • 如果需要,您可以使用一个数据库表来存储全局变量的值在 destroy() 中。在下次初始化 Servlet 时,该值可在 init() 方法内被读取。这一步是可选的。
  • 如果您只想对一个 session 会话计数一次页面点击,那么请使用 isNew() 方法来检查该 session 会话是否已点击过相同页面。这一步是可选的。
  • 您可以通过显示全局计数器的值,来在网站上展示页面的总点击量。这一步是可选的。

在这里,我们假设 Web 容器将无法重新启动。如果是重新启动或 Servlet 被销毁,计数器将被重置。

实例

本实例演示了如何实现一个简单的网页点击计数器:

import java.io.*;import java.sql.Date;import java.util.*;import javax.servlet.*;import javax.servlet.http.*;public class PageHitCounter extends HttpServlet{      private int hitCount;                  public void init()   {      // 重置点击计数器     hitCount = 0;  }   public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 设置响应内容类型      response.setContentType("text/html");      // 该方法在 Servlet 被点击时执行       // 增加 hitCount       hitCount++;       PrintWriter out = response.getWriter();      String title = "总点击量";      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +        "<html>
" +        "<head><title>" + title + "</title></head>
" +        "<body bgcolor="#f0f0f0">
" +        "<h1 align="center">" + title + "</h1>
" +        "<h2 align="center">" + hitCount + "</h2>
" +        "</body></html>");  }  public void destroy()   {       // 这一步是可选的,但是如果需要,您可以把 hitCount 的值写入到数据库  } } 

现在让我们来编译上面的 Servlet,并在 web.xml 文件中创建以下条目:

.... <servlet>     <servlet-name>PageHitCounter</servlet-name>     <servlet-class>PageHitCounter</servlet-class> </servlet> <servlet-mapping>     <servlet-name>PageHitCounter</servlet-name>     <url-pattern>/PageHitCounter</url-pattern> </servlet-mapping>....

现在通过访问 URL http://localhost:8080/PageHitCounter 来调用这个 Servlet。这将会在每次页面刷新时,把计数器的值增加 1,结果如下所示:

总点击量

6


网站点击计数器

很多时候,您可能有兴趣知道整个网站的总点击量。在 Servlet 中,这也是非常简单的,我们可以使用过滤器做到这一点。

以下是实现一个简单的基于过滤器生命周期的网站点击计数器需要采取的步骤:

  • 在过滤器的 init() 方法中初始化一个全局变量。
  • 每次调用 doFilter 方法时,都增加全局变量。
  • 如果需要,您可以使用一个数据库表来存储全局变量的值在过滤器的 destroy() 中。在下次初始化过滤器时,该值可在 init() 方法内被读取。这一步是可选的。

在这里,我们假设 Web 容器将无法重新启动。如果是重新启动或 Servlet 被销毁,点击计数器将被重置。

实例

本实例演示了如何实现一个简单的网站点击计数器:

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*;import java.util.*;public class SiteHitCounter implements Filter{      private int hitCount;                  public void  init(FilterConfig config)                     throws ServletException{     // 重置点击计数器     hitCount = 0;  }  public void  doFilter(ServletRequest request,               ServletResponse response,              FilterChain chain)               throws java.io.IOException, ServletException {      // 把计数器的值增加 1      hitCount++;      // 输出计数器      System.out.println("网站访问统计:"+ hitCount );      // 把请求传回到过滤器链      chain.doFilter(request,response);  }  public void destroy()   {       // 这一步是可选的,但是如果需要,您可以把 hitCount 的值写入到数据库  } } 

现在让我们来编译上面的 Servlet,并在 web.xml 文件中创建以下条目:

....<filter>   <filter-name>SiteHitCounter</filter-name>   <filter-class>SiteHitCounter</filter-class></filter><filter-mapping>   <filter-name>SiteHitCounter</filter-name>   <url-pattern>/*</url-pattern></filter-mapping>....

现在访问网站的任意页面,比如 http://localhost:8080/。这将会在每次任意页面被点击时,把计数器的值增加 1,它会在日志中显示以下消息:

网站访问统计: 1网站访问统计: 2网站访问统计: 3网站访问统计: 4网站访问统计: 5..................

Servlet 自动刷新页面

假设有一个网页,它是显示现场比赛成绩或股票市场状况或货币兑换率。对于所有这些类型的页面,您需要定期刷新网页。

Java Servlet 提供了一个机制,使得网页会在给定的时间间隔自动刷新。

刷新网页的最简单的方式是使用响应对象的方法 setIntHeader()。以下是这种方法的定义:

public void setIntHeader(String header, int headerValue)

此方法把头信息 "Refresh" 连同一个表示时间间隔的整数值(以秒为单位)发送回浏览器。

自动刷新页面实例

本实例演示了 Servlet 如何使用 setIntHeader() 方法来设置 Refresh 头信息,从而实现自动刷新页面。

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*;import java.util.*; // 扩展 HttpServlet 类public class Refresh extends HttpServlet {   // 处理 GET 方法请求的方法  public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 设置刷新自动加载的事件间隔为 5 秒      response.setIntHeader("Refresh", 5);       // 设置响应内容类型      response.setContentType("text/html");       // 获取当前的时间      Calendar calendar = new GregorianCalendar();      String am_pm;      int hour = calendar.get(Calendar.HOUR);      int minute = calendar.get(Calendar.MINUTE);      int second = calendar.get(Calendar.SECOND);      if(calendar.get(Calendar.AM_PM) == 0)        am_pm = "AM";      else        am_pm = "PM";       String CT = hour+":"+ minute +":"+ second +" "+ am_pm;          PrintWriter out = response.getWriter();      String title = "使用 Servlet 自动刷新页面";      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +        "<html>
" +        "<head><title>" + title + "</title></head>
"+        "<body bgcolor="#f0f0f0">
" +        "<h1 align="center">" + title + "</h1>
" +        "<p>当前时间是:" + CT + "</p>
");  }  // 处理 POST 方法请求的方法  public void doPost(HttpServletRequest request,                     HttpServletResponse response)      throws ServletException, IOException {     doGet(request, response);  }}

现在让我们来编译上面的 Servlet,并在 web.xml 文件中创建以下条目:

.... <servlet>     <servlet-name>Refresh</servlet-name>     <servlet-class>Refresh</servlet-class> </servlet>  <servlet-mapping>     <servlet-name>Refresh</servlet-name>     <url-pattern>/Refresh</url-pattern> </servlet-mapping>....

现在通过访问 URL http://localhost:8080/Refresh 来调用这个 Servlet。这将会每隔 5 秒钟显示一次当前系统时间。运行该 Servlet,并等待查看结果:

使用 Servlet 自动刷新页面

当前时间是:9:44:50 PM

Servlet 发送电子邮件

使用 Servlet 发送一封电子邮件是很简单的,但首先您必须在您的计算机上安装 JavaMail APIJava Activation Framework)JAF)

下载并解压缩这些文件,在新创建的顶层目录中,您会发现这两个应用程序的一些 jar 文件。您需要把 mail.jaractivation.jar 文件添加到您的 CLASSPATH 中。

发送一封简单的电子邮件

下面的实例将从您的计算机上发送一封简单的电子邮件。这里假设您的本地主机已连接到互联网,并支持发送电子邮件。同时确保 Java Email API 包和 JAF 包的所有的 jar 文件在 CLASSPATH 中都是可用的。

// 文件名 SendEmail.javaimport java.io.*;import java.util.*;import javax.servlet.*;import javax.servlet.http.*;import javax.mail.*;import javax.mail.internet.*;import javax.activation.*; public class SendEmail extends HttpServlet{      public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 收件人的电子邮件 ID      String to = "abcd@gmail.com";       // 发件人的电子邮件 ID      String from = "web@gmail.com";       // 假设您是从本地主机发送电子邮件      String host = "localhost";       // 获取系统的属性      Properties properties = System.getProperties();       // 设置邮件服务器      properties.setProperty("mail.smtp.host", host);       // 获取默认的 Session 对象      Session session = Session.getDefaultInstance(properties);         // 设置响应内容类型      response.setContentType("text/html");      PrintWriter out = response.getWriter();      try{         // 创建一个默认的 MimeMessage 对象         MimeMessage message = new MimeMessage(session);         // 设置 From: header field of the header.         message.setFrom(new InternetAddress(from));         // 设置 To: header field of the header.         message.addRecipient(Message.RecipientType.TO,                                  new InternetAddress(to));         // 设置 Subject: header field         message.setSubject("This is the Subject Line!");         // 现在设置实际消息         message.setText("This is actual message");         // 发送消息         Transport.send(message);         String title = "发送电子邮件";         String res = "成功发送消息...";         String docType =         "<!doctype html public "-//w3c//dtd html 4.0 " +          "transitional//en">
";         out.println(docType +         "<html>
" +         "<head><title>" + title + "</title></head>
" +         "<body bgcolor="#f0f0f0">
" +         "<h1 align="center">" + title + "</h1>
" +         "<p align="center">" + res + "</p>
" +         "</body></html>");      }catch (MessagingException mex) {         mex.printStackTrace();      }   }} 

现在让我们来编译上面的 Servlet,并在 web.xml 文件中创建以下条目:

.... <servlet>     <servlet-name>SendEmail</servlet-name>     <servlet-class>SendEmail</servlet-class> </servlet>  <servlet-mapping>     <servlet-name>SendEmail</servlet-name>     <url-pattern>/SendEmail</url-pattern> </servlet-mapping>....

现在通过访问 URL http://localhost:8080/SendEmail 来调用这个 Servlet。这将会发送一封电子邮件到给定的电子邮件 ID abcd@gmail.com,并将显示下面所示的响应:

发送电子邮件

成功发送消息...

如果您想要发送一封电子邮件给多个收件人,那么请使用下面的方法来指定多个电子邮件 ID:

void addRecipients(Message.RecipientType type,                    Address[] addresses)throws MessagingException

下面是对参数的描述:

  • type:这将被设置为 TO、CC 或 BCC。在这里,CC 代表抄送,BCC 代表密件抄送。例如 Message.RecipientType.TO
  • addresses:这是电子邮件 ID 的数组。当指定电子邮件 ID 时,您需要使用 InternetAddress() 方法。

发送一封 HTML 电子邮件

下面的实例将从您的计算机上发送一封 HTML 格式的电子邮件。这里假设您的本地主机已连接到互联网,并支持发送电子邮件。同时确保 Java Email API 包和 JAF 包的所有的 jar 文件在 CLASSPATH 中都是可用的。

本实例与上一个实例很类似,但是这里我们使用 setContent() 方法来设置第二个参数为 "text/html" 的内容,该参数用来指定 HTML 内容是包含在消息中的。

使用这个实例,您可以发送内容大小不限的 HTML 内容。

// 文件名 SendEmail.javaimport java.io.*;import java.util.*;import javax.servlet.*;import javax.servlet.http.*;import javax.mail.*;import javax.mail.internet.*;import javax.activation.*; public class SendEmail extends HttpServlet{      public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 收件人的电子邮件 ID      String to = "abcd@gmail.com";       // 发件人的电子邮件 ID      String from = "web@gmail.com";       // 假设您是从本地主机发送电子邮件      String host = "localhost";       // 获取系统的属性      Properties properties = System.getProperties();       // 设置邮件服务器      properties.setProperty("mail.smtp.host", host);       // 获取默认的 Session 对象      Session session = Session.getDefaultInstance(properties);            // 设置响应内容类型      response.setContentType("text/html");      PrintWriter out = response.getWriter();      try{         // 创建一个默认的 MimeMessage 对象         MimeMessage message = new MimeMessage(session);         // 设置 From: header field of the header.         message.setFrom(new InternetAddress(from));         // 设置 To: header field of the header.         message.addRecipient(Message.RecipientType.TO,                                  new InternetAddress(to));         // 设置 Subject: header field         message.setSubject("This is the Subject Line!");         // 设置实际的 HTML 消息,内容大小不限         message.setContent("<h1>This is actual message</h1>",                            "text/html" );         // 发送消息         Transport.send(message);         String title = "发送电子邮件";         String res = "成功发送消息...";         String docType =         "<!doctype html public "-//w3c//dtd html 4.0 " +          "transitional//en">
";         out.println(docType +         "<html>
" +         "<head><title>" + title + "</title></head>
" +         "<body bgcolor="#f0f0f0">
" +         "<h1 align="center">" + title + "</h1>
" +         "<p align="center">" + res + "</p>
" +         "</body></html>");      }catch (MessagingException mex) {         mex.printStackTrace();      }   }} 

编译并运行上面的 Servlet ,在给定的电子邮件 ID 上发送 HTML 消息。

在电子邮件中发送附件

下面的实例将从您的计算机上发送一封带有附件的电子邮件。这里假设您的本地主机已连接到互联网,并支持发送电子邮件。同时确保 Java Email API 包和 JAF 包的所有的 jar 文件在 CLASSPATH 中都是可用的。

// 文件名 SendEmail.javaimport java.io.*;import java.util.*;import javax.servlet.*;import javax.servlet.http.*;import javax.mail.*;import javax.mail.internet.*;import javax.activation.*; public class SendEmail extends HttpServlet{      public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 收件人的电子邮件 ID      String to = "abcd@gmail.com";       // 发件人的电子邮件 ID      String from = "web@gmail.com";       // 假设您是从本地主机发送电子邮件      String host = "localhost";       // 获取系统的属性      Properties properties = System.getProperties();       // 设置邮件服务器      properties.setProperty("mail.smtp.host", host);       // 获取默认的 Session 对象      Session session = Session.getDefaultInstance(properties);           // 设置响应内容类型      response.setContentType("text/html");      PrintWriter out = response.getWriter();       try{         // 创建一个默认的 MimeMessage 对象         MimeMessage message = new MimeMessage(session);          // 设置 From: header field of the header.         message.setFrom(new InternetAddress(from));          // 设置 To: header field of the header.         message.addRecipient(Message.RecipientType.TO,                                  new InternetAddress(to));          // 设置 Subject: header field         message.setSubject("This is the Subject Line!");          // 创建消息部分          BodyPart messageBodyPart = new MimeBodyPart();          // 填写消息         messageBodyPart.setText("This is message body");                  // 创建一个多部分消息         Multipart multipart = new MimeMultipart();          // 设置文本消息部分         multipart.addBodyPart(messageBodyPart);          // 第二部分是附件         messageBodyPart = new MimeBodyPart();         String filename = "file.txt";         DataSource source = new FileDataSource(filename);         messageBodyPart.setDataHandler(new DataHandler(source));         messageBodyPart.setFileName(filename);         multipart.addBodyPart(messageBodyPart);          // 发送完整的消息部分         message.setContent(multipart );          // 发送消息         Transport.send(message);         String title = "发送电子邮件";         String res = "成功发送电子邮件...";         String docType =         "<!doctype html public "-//w3c//dtd html 4.0 " +          "transitional//en">
";         out.println(docType +         "<html>
" +         "<head><title>" + title + "</title></head>
" +         "<body bgcolor="#f0f0f0">
" +         "<h1 align="center">" + title + "</h1>
" +         "<p align="center">" + res + "</p>
" +         "</body></html>");      }catch (MessagingException mex) {         mex.printStackTrace();      }   }} 

编译并运行上面的 Servlet ,在给定的电子邮件 ID 上发送带有文件附件的消息。

用户身份认证部分

如果需要向电子邮件服务器提供用户 ID 和密码进行身份认证,那么您可以设置如下属性:

 props.setProperty("mail.user", "myuser"); props.setProperty("mail.password", "mypwd");

电子邮件发送机制的其余部分与上面讲解的保持一致。

Servlet

涉及到 WEB-INF 子目录的 Web 应用程序结构是所有的 Java web 应用程序的标准,并由 Servlet API 规范指定。给定一个顶级目录名 myapp,目录结构如下所示:

/myapp    /images    /WEB-INF        /classes        /lib

WEB-INF 子目录中包含应用程序的部署描述符,名为 web.xml。所有的 HTML 文件都位于顶级目录 myapp 下。对于 admin 用户,您会发现 ROOT 目录是 myApp 的父目录。

创建包中的 Servlet

WEB-INF/classes 目录包含了所有的 Servlet 类和其他类文件,类文件所在的目录结构与他们的包名称匹配。例如,如果您有一个完全合格的类名称 com.myorg.MyServlet,那么这个 Servlet 类必须位于以下目录中:

/myapp/WEB-INF/classes/com/myorg/MyServlet.class

下面的例子创建包名为 com.myorg 的 MyServlet 类。

// 为包命名package com.myorg;  // 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*; public class MyServlet extends HttpServlet {   private String message;   public void init() throws ServletException  {      // 执行必需的的初始化      message = "Hello World";  }   public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 设置响应内容类型      response.setContentType("text/html");       // 实际的逻辑是在这里      PrintWriter out = response.getWriter();      out.println("<h1>" + message + "</h1>");  }    public void destroy()  {      // 什么也不做  }}

编译包中的 Servlet

编译包中的类与编译其他的类没有什么大的不同。最简单的方法是让您的 java 文件保留完全限定路径,如上面提到的类,将被保留在 com.myorg 中。您还需要在 CLASSPATH 中添加该目录。

假设您的环境已正确设置,进入 <Tomcat-installation-directory>/webapps/ROOT/WEB-INF/classes 目录,并编译 MyServlet.java,如下所示:

$ javac MyServlet.java

如果 Servlet 依赖于其他库,那么您必须在 CLASSPATH 中也要引用那些 JAR 文件。这里我只引用了 servlet-api.jar JAR 文件,因为我在 Hello World 程序中并没有使用任何其他库。

该命令行使用内置的 javac 编译器,它是 Sun Microsystems Java 软件开发工具包(JDK,全称 Java Software Development Kit)附带的。 Microsystems的Java软件开发工具包(JDK)。为了让该命令正常工作,必须包括您在 PATH 环境变量中所使用的 Java SDK 的位置。

如果一切顺利,上述编译会在同一目录下生成 MyServlet.class 文件。下一节将解释如何把一个已编译的 Servlet 部署到生产中。

Servlet 打包部署

默认情况下,Servlet 应用程序位于路径 <Tomcat-installation-directory>/webapps/ROOT 下,且类文件放在 <Tomcat-installation-directory>/webapps/ROOT/WEB-INF/classes 中。

如果您有一个完全合格的类名称 com.myorg.MyServlet,那么这个 Servlet 类必须位于 WEB-INF/classes/com/myorg/MyServlet.class 中,您需要在位于 <Tomcat-installation-directory>/webapps/ROOT/WEB-INF/ 的 web.xml 文件中创建以下条目:

    <servlet>        <servlet-name>MyServlet</servlet-name>        <servlet-class>com.myorg.MyServlet</servlet-class>    </servlet>     <servlet-mapping>        <servlet-name>MyServlet</servlet-name>        <url-pattern>/MyServlet</url-pattern>    </servlet-mapping>

上面的条目要被创建在 web.xml 文件中的 <web-app>...</web-app> 标签内。在该文件中可能已经有各种可用的条目,但不要在意。

到这里,您基本上已经完成了,现在让我们使用 <Tomcat-installation-directory>instartup.bat(在 Windows 上)或 <Tomcat-installation-directory>/bin/startup.sh(在 Linux/Solaris 等上)启动 tomcat 服务器,最后在浏览器的地址栏中输入 http://localhost:8080/MyServlet。如果一切顺利,您会看到下面的结果:

Hello World

Servlet 调试

测试/调试 Servlet 始终是开发使用过程中的难点。Servlet 往往涉及大量的客户端/服务器交互,可能会出现错误但又难以重现。

这里有一些提示和建议,可以帮助您调试。

System.out.println()

System.out.println() 是作为一个标记来使用的,用来测试一段特定的代码是否被执行。我们也可以打印出变量的值。此外:

  • 由于 System 对象是核心 Java 对象的一部分,它可以在不需要安装任何额外类的情况下被用于任何地方。这包括 Servlet、JSP、RMI、EJB's、普通的 Beans 和类,以及独立的应用程序。
  • 与在断点处停止不同,写入到 System.out 不会干扰到应用程序的正常执行流程,这使得它在时序是至关重要的时候显得尤为有价值。

下面是使用 System.out.println() 的语法:

System.out.println("Debugging message");

通过上面的语法生成的所有消息将被记录在 Web 服务器日志文件中。

消息日志

使用适当的日志记录方法来记录所有调试、警告和错误消息,这是非常好的想法,推荐使用 log4J 来记录所有的消息。

Servlet API 还提供了一个简单的输出信息的方式,使用 log() 方法,如下所示:

// 导入必需的 java 库import java.io.*;import javax.servlet.*;import javax.servlet.http.*;public class ContextLog extends HttpServlet {  public void doGet(HttpServletRequest request,       HttpServletResponse response) throws ServletException,         java.io.IOException {          String par = request.getParameter("par1");      // 调用两个 ServletContext.log 方法      ServletContext context = getServletContext( );      if (par == null || par.equals(""))      // 通过 Throwable 参数记录版本      context.log("No message received:",          new IllegalStateException("Missing parameter"));      else          context.log("Here is the visitor's message: " + par);            response.setContentType("text/html");      java.io.PrintWriter out = response.getWriter( );      String title = "Context Log";      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +        "<html>
" +        "<head><title>" + title + "</title></head>
" +        "<body bgcolor="#f0f0f0">
" +        "<h1 align="center">" + title + "</h1>
" +        "<h2 align="center">Messages sent</h2>
" +        "</body></html>");    } //doGet}

ServletContext 把它的文本消息记录到 Servlet 容器的日志文件中。对于 Tomcat,这些日志可以在 <Tomcat-installation-directory>/logs 目录中找到。

这些日志文件确实对新出现的错误或问题的频率给出指示。正因为如此,建议在通常不会发生的异常的 catch 子句中使用 log() 函数。

使用 JDB 调试器

您可以使用调试 applet 或应用程序的 jdb 命令来调试 Servlet。

为了调试一个 Servlet,我们可以调试 sun.servlet.http.HttpServer,然后把它看成是 HttpServer 执行 Servlet 来响应浏览器端的 HTTP 请求。这与调试 applet 小程序非常相似。与调试 applet 不同的是,实际被调试的程序是 sun.applet.AppletViewer。

大多数调试器会自动隐藏如何调试 applet 的细节。同样的,对于 servlet,您必须帮调试器执行以下操作:

  • 设置您的调试器的类路径 classpath,以便它可以找到 sun.servlet.http.Http-Server 和相关的类。
  • 设置您的调试器的类路径 classpath,以便它可以找到您的 servlet 和支持的类,通常是在 server_root/servlets 和 server_root/classes 中。

您通常不会希望 server_root/servlets 在您的 classpath 中,因为它会禁用 servlet 的重新加载。但是这种包含规则对于调试是非常有用的。它允许您的调试器在 HttpServer 中的自定义 Servlet 加载器加载 Servlet 之前在 Servlet 中设置断点。

如果您已经设置了正确的类路径 classpath,就可以开始调试 sun.servlet.http.HttpServer。可以在您想要调试的 Servlet 代码中设置断点,然后通过 Web 浏览器使用给定的 Servlet(http://localhost:8080/servlet/ServletToDebug)向 HttpServer 发出请求。您会看到程序执行到断点处会停止。

使用注释

代码中的注释有助于以各种方式进行调试。注释可用于调试过程的很多其他方式中。

该 Servlet 使用 Java 注释和单行注释(//...),多行注释(/* ...*/)可用于暂时移除部分 Java 代码。如果 bug 消失,仔细看看您刚才注释的代码并找出问题所在。

客户端和服务器端头信息

有时,当一个 Servlet 并没有像预期那样时,查看原始的 HTTP 请求和响应是非常有用的。如果您熟悉 HTTP 结构,您可以阅读请求和响应,看看这些头信息究竟是什么。

重要的调试技巧

下面列出了一些 Servlet 调试的技巧:

  • 请注意,server_root/classes 不会重载,而 server_root/servlets 可能会。
  • 要求浏览器显示它所显示的页面的原始内容。这有助于识别格式的问题。它通常是"视图"菜单下的一个选项。
  • 通过强制执行完全重新加载页面来确保浏览器还没有缓存前一个请求的输出。在 Netscape Navigator 中,请使用 Shift-Reload,在 Internet Explorer 中,请使用 Shift-Refresh。
  • 请确认 servlet 的 init() 方法接受一个 ServletConfig 参数,并调用 super.init(config)。

Servlet 国际化

在我们开始之前,先来看看三个重要术语:

  • 国际化(i18n):这意味着一个网站提供了不同版本的翻译成访问者的语言或国籍的内容。
  • 本地化(l10n):这意味着向网站添加资源,以使其适应特定的地理或文化区域,例如网站翻译成印地文(Hindi)。
  • 区域设置(locale):这是一个特殊的文化或地理区域。它通常指语言符号后跟一个下划线和一个国家符号。例如 "en_US" 表示针对 US 的英语区域设置。

当建立一个全球性的网站时有一些注意事项。本教程不会讲解这些注意事项的完整细节,但它会通过一个很好的实例向您演示如何通过差异化定位(即区域设置)来让网页以不同语言呈现。

Servlet 可以根据请求者的区域设置拾取相应版本的网站,并根据当地的语言、文化和需求提供相应的网站版本。以下是 request 对象中返回 Locale 对象的方法。

java.util.Locale request.getLocale() 

检测区域设置

下面列出了重要的区域设置方法,您可以使用它们来检测请求者的地理位置、语言和区域设置。下面所有的方法都显示了请求者浏览器中设置的国家名称和语言名称。

序号方法 & 描述
1String getCountry()
该方法以 2 个大写字母形式的 ISO 3166 格式返回该区域设置的国家/地区代码。
2String getDisplayCountry()
该方法返回适合向用户显示的区域设置的国家的名称。
3String getLanguage()
该方法以小写字母形式的 ISO 639 格式返回该区域设置的语言代码。
4String getDisplayLanguage()
该方法返回适合向用户显示的区域设置的语言的名称。
5String getISO3Country()
该方法返回该区域设置的国家的三个字母缩写。
6String getISO3Language()
该方法返回该区域设置的语言的三个字母的缩写。

实例

本实例演示了如何显示某个请求的语言和相关的国家:

import java.io.*;import javax.servlet.*;import javax.servlet.http.*;import java.util.Locale;public class GetLocale extends HttpServlet{      public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {      // 获取客户端的区域设置      Locale locale = request.getLocale();      String language = locale.getLanguage();      String country = locale.getCountry();      // 设置响应内容类型      response.setContentType("text/html");      PrintWriter out = response.getWriter();      String title = "检测区域设置";      String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +        "<html>
" +        "<head><title>" + title + "</title></head>
" +        "<body bgcolor="#f0f0f0">
" +        "<h1 align="center">" + language + "</h1>
" +        "<h2 align="center">" + country + "</h2>
" +        "</body></html>");  }} 

语言设置

Servlet 可以输出以西欧语言(如英语、西班牙语、德语、法语、意大利语、荷兰语等)编写的页面。在这里,为了能正确显示所有的字符,设置 Content-Language 头是非常重要的。

第二点是使用 HTML 实体显示所有的特殊字符,例如,"ñ" 表示 "ñ","¡" 表示 "¡",如下所示:

import java.io.*;import javax.servlet.*;import javax.servlet.http.*;import java.util.Locale;public class DisplaySpanish extends HttpServlet{      public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {    // 设置响应内容类型    response.setContentType("text/html");    PrintWriter out = response.getWriter();    // 设置西班牙语言代码    response.setHeader("Content-Language", "es");    String title = "En Espa&ntilde;ol";    String docType =     "<!doctype html public "-//w3c//dtd html 4.0 " +      "transitional//en">
";     out.println(docType +     "<html>
" +     "<head><title>" + title + "</title></head>
" +     "<body bgcolor="#f0f0f0">
" +     "<h1>" + "En Espa&ntilde;ol:" + "</h1>
" +     "<h1>" + "&iexcl;Hola Mundo!" + "</h1>
" +     "</body></html>");  }} 

特定于区域设置的日期

您可以使用 java.text.DateFormat 类及其静态方法 getDateTimeInstance() 来格式化特定于区域设置的日期和时间。下面的实例演示了如何格式化特定于某个给定的区域设置的日期:

import java.io.*;import javax.servlet.*;import javax.servlet.http.*;import java.util.Locale;import java.text.DateFormat;import java.util.Date;public class DateLocale extends HttpServlet{      public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {    // 设置响应内容类型    response.setContentType("text/html");    PrintWriter out = response.getWriter();    // 获取客户端的区域设置    Locale locale = request.getLocale( );    String date = DateFormat.getDateTimeInstance(                                  DateFormat.FULL,                                   DateFormat.SHORT,                                   locale).format(new Date( ));    String title = "特定于区域设置的日期";    String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +      "<html>
" +      "<head><title>" + title + "</title></head>
" +      "<body bgcolor="#f0f0f0">
" +      "<h1 align="center">" + date + "</h1>
" +      "</body></html>");  }} 

特定于区域设置的货币

您可以使用 java.text.NumberFormat 类及其静态方法 getCurrencyInstance() 来格式化数字(比如 long 类型或 double 类型)为特定于区域设置的货币。下面的实例演示了如何格式化特定于某个给定的区域设置的货币:

import java.io.*;import javax.servlet.*;import javax.servlet.http.*;import java.util.Locale;import java.text.NumberFormat;import java.util.Date;public class CurrencyLocale extends HttpServlet{      public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {    // 设置响应内容类型    response.setContentType("text/html");    PrintWriter out = response.getWriter();    // 获取客户端的区域设置    Locale locale = request.getLocale( );    NumberFormat nft = NumberFormat.getCurrencyInstance(locale);    String formattedCurr = nft.format(1000000);    String title = "特定于区域设置的货币";    String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +      "<html>
" +      "<head><title>" + title + "</title></head>
" +      "<body bgcolor="#f0f0f0">
" +      "<h1 align="center">" + formattedCurr + "</h1>
" +      "</body></html>");  }} 

特定于区域设置的百分比

您可以使用 java.text.NumberFormat 类及其静态方法 getPercentInstance() 来格式化特定于区域设置的百分比。下面的实例演示了如何格式化特定于某个给定的区域设置的百分比:

import java.io.*;import javax.servlet.*;import javax.servlet.http.*;import java.util.Locale;import java.text.NumberFormat;import java.util.Date;public class PercentageLocale extends HttpServlet{      public void doGet(HttpServletRequest request,                    HttpServletResponse response)            throws ServletException, IOException  {    // 设置响应内容类型    response.setContentType("text/html");    PrintWriter out = response.getWriter();    // 获取客户端的区域设置    Locale locale = request.getLocale( );    NumberFormat nft = NumberFormat.getPercentInstance(locale);    String formattedPerc = nft.format(0.51);    String title = "特定于区域设置的百分比";    String docType =      "<!doctype html public "-//w3c//dtd html 4.0 " +       "transitional//en">
";      out.println(docType +      "<html>
" +      "<head><title>" + title + "</title></head>
" +      "<body bgcolor="#f0f0f0">
" +      "<h1 align="center">" + formattedPerc + "</h1>
" +      "</body></html>");  }} 

Servlet容器主要是JavaWeb应用提供运行时环境,所以也可以称之为JavaWeb应用容器,或者Servlet/JSP容器。Servlet容器主要负责管理Servlet、JSP的生命周期以及它们的共享数据。


Servlet容器有哪些:

目前最流行的Servlet容器软件包括: Tomcat、Jetty、Jboss等。


Tomcat

Tomcat和IIS等Web服务器一样,具有处理HTML页面的功能,另外它还是一个Servlet和JSP容器,独立的Servlet容器是Tomcat的默认模式。不过,Tomcat处理静态HTML的能力不如Apache服务器。


Jetty

Jetty 是一个开源的servlet容器,它为基于Java的web容器,例如JSP和servlet提供运行环境。Jetty是使用Java语言编写的,它的API以一组JAR包的形式发布。开发人员可以将Jetty容器实例化成一个对象,可以迅速为一些独立运行(stand-alone)的Java应用提供网络和web连接。


Jboss

Jboss是一个基于J2EE的开放源代码的应用服务器。 JBoss代码遵循LGPL许可,可以在任何商业应用中免费使用。JBoss是一个管理EJB的容器和服务器,支持EJB 1.1、EJB 2.0和EJB3的规范。但JBoss核心服务不包括支持servlet/JSP的WEB容器,一般与Tomcat或Jetty绑定使用。


Servlet是和平台无关的服务器端组件(java编写的,跨平台),它运行在Servlet容器中。

Servlet容器负责Servlet和客户的通信以及调用Servlet的方法,Servlet和客户的通信采用“请求/响应”的模式 Servlet可完成如下功能:

1、创建并返回基于客户请求的动态HTML页面

2、创建可嵌入到现有HTML 页面中的部分HTML 页面(HTML 片段)

3、与其它服务器资源(如数据库或基于Java的应用程序)进行通信


Servlet容器响应客户请求过程:

215229_9PD7_3839775


jsp作为Servlet技术的扩展,经常会有人将jsp和Servlet搞混。本文,将为大家带来servlet和jsp的区别,希望对大家有所帮助。


servlet和jsp的区别

1、Servlet在Java代码中可以通过HttpServletResponse对象动态输出HTML内容。

2、JSP是在静态HTML内容中嵌入Java代码,然后Java代码在被动态执行后生成HTML内容。


servlet和jsp各自的特点

1、Servlet虽然能够很好地组织业务逻辑代码,但是在Java源文件中,因为是通过字符串拼接的方式生成动态HTML内容,这样就容易导致代码维护困难、可读性差。

2、JSP虽然规避了Servlet在生成HTML内容方面的劣势,但是在HTML中混入大量、复杂的业务逻辑。


通过MVC双剑合璧

JSP和Servlet都有自身的适用环境,那么有没有什么办法能够让它们发挥各自的优势呢?答案是肯有的,MVC模式就能够完美解决这一问题。

MVC模式,是Model-View-Controller的简称,是软件工程中的一种软件架构模式,分为三个基本部分,分别是:模型(Model)、视图(View)和控制器(Controller):

Controller——负责转发请求,对请求进行处理

View——负责界面显示

Model——业务功能编写(例如算法实现)、数据库设计以及数据存取操作实现

在JSP/Servlet开发的软件系统中,这三个部分的描述如下所示:


229cf9ff5b1729eaf408fac56238eeb3_hd


1、Web浏览器发送HTTP请求到服务端,然后被Controller(Servlet)获取并进行处理(例如参数解析、请求转发)

2、Controller(Servlet)调用核心业务逻辑——Model部分,获得结果

3、Controller(Servlet)将逻辑处理结果交给View(JSP),动态输出HTML内容

4、动态生成的HTML内容返回到浏览器显示

MVC模式在Web开发中有很大的优势,它完美规避了JSP与Servlet各自的缺点,让Servlet只负责业务逻辑部分,而不会生成HTML代码;同时JSP中也不会充斥着大量的业务代码,这样能大提高了代码的可读性和可维护性。


相关阅读:

Servlet教程

JSP教程


Servlet 文件上传

Servlet 可以与 HTML form 标签一起使用,来允许用户上传文件到服务器。上传的文件可以是文本文件或图像文件或任何文档。

创建一个文件上传表单

下面的 HTML 代码创建了一个文件上传表单。以下几点需要注意:

  • 表单 method 属性应该设置为 POST 方法,不能使用 GET 方法。
  • 表单 enctype 属性应该设置为 multipart/form-data.
  • 表单 action 属性应该设置为在后端服务器上处理文件上传的 Servlet 文件。下面的实例使用了 UploadServlet Servlet 来上传文件。
  • 上传单个文件,您应该使用单个带有属性 type="file" 的 <input .../> 标签。为了允许多个文件上传,请包含多个 name 属性值不同的 input 标签。输入标签具有不同的名称属性的值。浏览器会为每个 input 标签关联一个浏览按钮。
<html><head><title>文件上传表单</title></head><body><h3>文件上传:</h3>请选择要上传的文件:<br /><form action="UploadServlet" method="post" enctype="multipart/form-data"><input type="file" name="file" size="50" /><br /><input type="submit" value="上传文件" /></form></body></html>

这将显示下面的结果,允许用户从本地计算机选择一个文件,当用户点击"上传文件"时,表单会连同从本地计算机选择的文件一起提交:

 <b>文件上传:</b> 请选择要上传的文件:<br /> <input type="file" name="file" size="50" /> <br /> <input type="button" value="上传文件" /> <br /> 注:这只是虚拟的表单,不会正常工作。

编写后台 Servlet

以下是 Servlet UploadServlet,会接受上传的文件,并把它储存在目录 <Tomcat-installation-directory>/webapps/data 中。这个目录名也可以使用外部配置来添加,比如 web.xml 中的 context-param 元素,如下所示:

<web-app>....<context-param>     <description>Location to store uploaded file</description>     <param-name>file-upload</param-name>     <param-value>         c:apache-tomcat-5.5.29webappsdata     </param-value> </context-param>....</web-app>

以下是 UploadServlet 的源代码,可以一次处理多个文件的上传。在继续操作之前,请确认下列各项:

  • 下面的实例依赖于 FileUpload,所以一定要确保在您的 classpath 中有最新版本的 commons-fileupload.x.x.jar 文件。可以从 http://commons.apache.org/fileupload/ 下载。
  • FileUpload 依赖于 Commons IO,所以一定要确保在您的 classpath 中有最新版本的 commons-io-x.x.jar 文件。可以从 http://commons.apache.org/io/ 下载。
  • 在测试下面实例时,您上传的文件大小不能大于 maxFileSize,否则文件将无法上传。
  • 请确保已经提前创建好目录 c: emp and c:apache-tomcat-5.5.29webappsdata。
// 导入必需的 java 库import java.io.*;import java.util.*; import javax.servlet.ServletConfig;import javax.servlet.ServletException;import javax.servlet.http.HttpServlet;import javax.servlet.http.HttpServletRequest;import javax.servlet.http.HttpServletResponse; import org.apache.commons.fileupload.FileItem;import org.apache.commons.fileupload.FileUploadException;import org.apache.commons.fileupload.disk.DiskFileItemFactory;import org.apache.commons.fileupload.servlet.ServletFileUpload;import org.apache.commons.io.output.*;public class UploadServlet extends HttpServlet {      private boolean isMultipart;   private String filePath;   private int maxFileSize = 50 * 1024;   private int maxMemSize = 4 * 1024;   private File file ;   public void init( ){      // 获取文件将被存储的位置      filePath =              getServletContext().getInitParameter("file-upload");    }   public void doPost(HttpServletRequest request,                HttpServletResponse response)              throws ServletException, java.io.IOException {      // 检查我们有一个文件上传请求      isMultipart = ServletFileUpload.isMultipartContent(request);      response.setContentType("text/html");      java.io.PrintWriter out = response.getWriter( );      if( !isMultipart ){         out.println("<html>");         out.println("<head>");         out.println("<title>Servlet upload</title>");           out.println("</head>");         out.println("<body>");         out.println("<p>No file uploaded</p>");          out.println("</body>");         out.println("</html>");         return;      }      DiskFileItemFactory factory = new DiskFileItemFactory();      // 文件大小的最大值将被存储在内存中      factory.setSizeThreshold(maxMemSize);      // Location to save data that is larger than maxMemSize.      factory.setRepository(new File("c:	emp"));      // 创建一个新的文件上传处理程序      ServletFileUpload upload = new ServletFileUpload(factory);      // 允许上传的文件大小的最大值      upload.setSizeMax( maxFileSize );      try{       // 解析请求,获取文件项      List fileItems = upload.parseRequest(request);         // 处理上传的文件项      Iterator i = fileItems.iterator();      out.println("<html>");      out.println("<head>");      out.println("<title>Servlet upload</title>");        out.println("</head>");      out.println("<body>");      while ( i.hasNext () )       {         FileItem fi = (FileItem)i.next();         if ( !fi.isFormField () )            {            // 获取上传文件的参数            String fieldName = fi.getFieldName();            String fileName = fi.getName();            String contentType = fi.getContentType();            boolean isInMemory = fi.isInMemory();            long sizeInBytes = fi.getSize();            // 写入文件            if( fileName.lastIndexOf("") >= 0 ){               file = new File( filePath +                fileName.substring( fileName.lastIndexOf(""))) ;            }else{               file = new File( filePath +                fileName.substring(fileName.lastIndexOf("")+1)) ;            }            fi.write( file ) ;            out.println("Uploaded Filename: " + fileName + "<br>");         }      }      out.println("</body>");      out.println("</html>");   }catch(Exception ex) {       System.out.println(ex);   }   }   public void doGet(HttpServletRequest request,                        HttpServletResponse response)        throws ServletException, java.io.IOException {                throw new ServletException("GET method used with " +                getClass( ).getName( )+": POST method required.");   } }

编译和运行 Servlet

编译上面的 Servlet UploadServlet,并在 web.xml 文件中创建所需的条目,如下所示:

<servlet>   <servlet-name>UploadServlet</servlet-name>   <servlet-class>UploadServlet</servlet-class></servlet><servlet-mapping>   <servlet-name>UploadServlet</servlet-name>   <url-pattern>/UploadServlet</url-pattern></servlet-mapping>

现在尝试使用您在上面创建的 HTML 表单来上传文件。当您在浏览器中访问:http://localhost:8080/UploadFile.htm 时,它会显示下面的结果,这将有助于您从本地计算机上传任何文件。

<b>文件上传:</b> 请选择要上传的文件:<br /> <input type="file" name="file" size="50" /> <br /> <input type="button" value="上传文件" /> 

如果您的 Servelt 脚本能正常工作,那么您的文件会被上传到 c:apache-tomcat-5.5.29webappsdata 目录中。

Servlet 文件上传

Servlet 可以与 HTML form 标签一起使用,来允许用户上传文件到服务器。上传的文件可以是文本文件或图像文件或任何文档。

创建一个文件上传表单

下面的 HTML 代码创建了一个文件上传表单。以下几点需要注意:

  • 表单 method 属性应该设置为 POST 方法,不能使用 GET 方法。
  • 表单 enctype 属性应该设置为 multipart/form-data.
  • 表单 action 属性应该设置为在后端服务器上处理文件上传的 Servlet 文件。下面的实例使用了 UploadServlet Servlet 来上传文件。
  • 上传单个文件,您应该使用单个带有属性 type="file" 的 <input .../> 标签。为了允许多个文件上传,请包含多个 name 属性值不同的 input 标签。输入标签具有不同的名称属性的值。浏览器会为每个 input 标签关联一个浏览按钮。
<html><head><title>文件上传表单</title></head><body><h3>文件上传:</h3>请选择要上传的文件:<br /><form action="UploadServlet" method="post" enctype="multipart/form-data"><input type="file" name="file" size="50" /><br /><input type="submit" value="上传文件" /></form></body></html>

这将显示下面的结果,允许用户从本地计算机选择一个文件,当用户点击"上传文件"时,表单会连同从本地计算机选择的文件一起提交:

 <b>文件上传:</b> 请选择要上传的文件:<br /> <input type="file" name="file" size="50" /> <br /> <input type="button" value="上传文件" /> <br /> 注:这只是虚拟的表单,不会正常工作。

编写后台 Servlet

以下是 Servlet UploadServlet,会接受上传的文件,并把它储存在目录 <Tomcat-installation-directory>/webapps/data 中。这个目录名也可以使用外部配置来添加,比如 web.xml 中的 context-param 元素,如下所示:

<web-app>....<context-param>     <description>Location to store uploaded file</description>     <param-name>file-upload</param-name>     <param-value>         c:apache-tomcat-5.5.29webappsdata     </param-value> </context-param>....</web-app>

以下是 UploadServlet 的源代码,可以一次处理多个文件的上传。在继续操作之前,请确认下列各项:

  • 下面的实例依赖于 FileUpload,所以一定要确保在您的 classpath 中有最新版本的 commons-fileupload.x.x.jar 文件。可以从 http://commons.apache.org/fileupload/ 下载。
  • FileUpload 依赖于 Commons IO,所以一定要确保在您的 classpath 中有最新版本的 commons-io-x.x.jar 文件。可以从 http://commons.apache.org/io/ 下载。
  • 在测试下面实例时,您上传的文件大小不能大于 maxFileSize,否则文件将无法上传。
  • 请确保已经提前创建好目录 c: emp and c:apache-tomcat-5.5.29webappsdata。
// 导入必需的 java 库import java.io.*;import java.util.*; import javax.servlet.ServletConfig;import javax.servlet.ServletException;import javax.servlet.http.HttpServlet;import javax.servlet.http.HttpServletRequest;import javax.servlet.http.HttpServletResponse; import org.apache.commons.fileupload.FileItem;import org.apache.commons.fileupload.FileUploadException;import org.apache.commons.fileupload.disk.DiskFileItemFactory;import org.apache.commons.fileupload.servlet.ServletFileUpload;import org.apache.commons.io.output.*;public class UploadServlet extends HttpServlet {      private boolean isMultipart;   private String filePath;   private int maxFileSize = 50 * 1024;   private int maxMemSize = 4 * 1024;   private File file ;   public void init( ){      // 获取文件将被存储的位置      filePath =              getServletContext().getInitParameter("file-upload");    }   public void doPost(HttpServletRequest request,                HttpServletResponse response)              throws ServletException, java.io.IOException {      // 检查我们有一个文件上传请求      isMultipart = ServletFileUpload.isMultipartContent(request);      response.setContentType("text/html");      java.io.PrintWriter out = response.getWriter( );      if( !isMultipart ){         out.println("<html>");         out.println("<head>");         out.println("<title>Servlet upload</title>");           out.println("</head>");         out.println("<body>");         out.println("<p>No file uploaded</p>");          out.println("</body>");         out.println("</html>");         return;      }      DiskFileItemFactory factory = new DiskFileItemFactory();      // 文件大小的最大值将被存储在内存中      factory.setSizeThreshold(maxMemSize);      // Location to save data that is larger than maxMemSize.      factory.setRepository(new File("c:	emp"));      // 创建一个新的文件上传处理程序      ServletFileUpload upload = new ServletFileUpload(factory);      // 允许上传的文件大小的最大值      upload.setSizeMax( maxFileSize );      try{       // 解析请求,获取文件项      List fileItems = upload.parseRequest(request);         // 处理上传的文件项      Iterator i = fileItems.iterator();      out.println("<html>");      out.println("<head>");      out.println("<title>Servlet upload</title>");        out.println("</head>");      out.println("<body>");      while ( i.hasNext () )       {         FileItem fi = (FileItem)i.next();         if ( !fi.isFormField () )            {            // 获取上传文件的参数            String fieldName = fi.getFieldName();            String fileName = fi.getName();            String contentType = fi.getContentType();            boolean isInMemory = fi.isInMemory();            long sizeInBytes = fi.getSize();            // 写入文件            if( fileName.lastIndexOf("") >= 0 ){               file = new File( filePath +                fileName.substring( fileName.lastIndexOf(""))) ;            }else{               file = new File( filePath +                fileName.substring(fileName.lastIndexOf("")+1)) ;            }            fi.write( file ) ;            out.println("Uploaded Filename: " + fileName + "<br>");         }      }      out.println("</body>");      out.println("</html>");   }catch(Exception ex) {       System.out.println(ex);   }   }   public void doGet(HttpServletRequest request,                        HttpServletResponse response)        throws ServletException, java.io.IOException {                throw new ServletException("GET method used with " +                getClass( ).getName( )+": POST method required.");   } }

编译和运行 Servlet

编译上面的 Servlet UploadServlet,并在 web.xml 文件中创建所需的条目,如下所示:

<servlet>   <servlet-name>UploadServlet</servlet-name>   <servlet-class>UploadServlet</servlet-class></servlet><servlet-mapping>   <servlet-name>UploadServlet</servlet-name>   <url-pattern>/UploadServlet</url-pattern></servlet-mapping>

现在尝试使用您在上面创建的 HTML 表单来上传文件。当您在浏览器中访问:http://localhost:8080/UploadFile.htm 时,它会显示下面的结果,这将有助于您从本地计算机上传任何文件。

<b>文件上传:</b> 请选择要上传的文件:<br /> <input type="file" name="file" size="50" /> <br /> <input type="button" value="上传文件" /> 

如果您的 Servelt 脚本能正常工作,那么您的文件会被上传到 c:apache-tomcat-5.5.29webappsdata 目录中。