Servlet API
Servlet的框架的核心是javax.servlet.Servlet接口,所有的Servlet都必须实现这一接口。在Servlet接口中定义了五个方法,其中有三个方法代表了Servlet的生命周期:
- init方法:负责初始化Servlet对象
- service方法:负责响应客户的请求
- destroy方法:当Servlet对象退出生命周期时,负责释放占用的资源。
换句话说,Servlet的生命周期可分为三个阶段:初始化阶段;响应客户请求阶段;终止阶段。
Servlet的初始化阶段:
在下列时刻Servlet容器装载Servlet:
- Servlet容器启动时自动装载某些Servlet
- 在Servlet容器启动后,客户首次向Servlet发出请求
- Servlet的类文件被更新后,重新装载Servlet
Servlet被装载后,Servlet容器创建一个Servlet实例并且调用Servlet的init()方法进行初始化。在Servlet的整个生命周期中,init方法只会被调用一次。
Servlet的响应客户请求阶段:
对于到达Servlet容器的客户请求,Servlet容器创建特定于这个请求的ServletRequest对象和ServletResponse对象,然后调用Servlet的service方法。service方法从ServletRequest对象获得客户请求信息、处理该请求,并通过ServletResponse对象向客户返回响应结果。
Servlet的终止阶段:
当Web应用被终止,或Servlet容器终止运行,或Servlet容器重新装载Servlet的新实例时,Servlet容器会先调用Servlet的destroy方法。在destroy方法中,可以释放Servlet所占用的资源。
Servlet API类图如下:
- 如果你的Servlet类扩展了HttpServlet类,你通常不必实现service方法,因为HttpServlet类已经实现了service方法,该方法的声明形式:protected void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException
- 在HttpServlet的service方法中,首先从HttpServletRequest对象中获取HTTP请求方式的信息,然后再根据请求方式调用相应的方法。例如,如果请求方式为GET,那么调用doGet方法;如果请求方式为POST,那么调用doPost方法。
Tomcat service源码伪码描述如下:
public void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { String method = request.getMethod(); if(method.equalsIgnoreCase("get")) { this.doGet(request, response); } else if(method.equalsIgnoreCase("post")) { this.doPost(request, response); } else if ... }
ServletRequest接口
- ServletRequest接口封装了客户请求信息,如客户请求方式、参数名和参数值、客户端正在使用的协议,以及发出客户请求的远程主机信息等。ServletRequest接口还为Servlet提供了直接以二进制方式读取客户请求数据流的ServletInputStream。
- ServletRequest的子类可以为Servlet提供更多的和特定协议相关的数据,例如:HttpServletRequest提供了读取HTTP Head信息的方法。
ServletRequest接口的主要方法
- getAttribute,根据参数给定的属性名返回属性值
- getContentType,返回客户请求数据MIME类型
- getInputStream,返回以二进制方式直接读取客户请求数据的输入流
- getParameter,根据给定的参数名返回参数值
- getRemoteAddr,返回远程客户主机的IP地址
- getRomoteHost,返回远程客户主机名
- getRemotePort,返回远程客户主机的端口
ServletResponse接口
- ServletResponse接口为Servlet提供了返回响应结果的方法。它允许Servlet设置返回数据的长度和MIME类型,并且提供输出流ServletOutputStream。
- ServletRequest的子类可以为Servlet提供更多的和特定协议相关的数据,例如:HttpServletResponse提供了设定HTTP Head信息的方法。
ServletResponse接口的主要方法
- getOutputStream,返回可以向客户端发送二进制数据的输出流对象ServletOutputStream
- getWriter,返回可以向客户端发送字符数据的PrintWrite对象
- getCharacterEncoding,返回Servlet发送的相应数据的字节编码
- getContentType,返回Servlet发送的响应数据的MIME类型
- setContentType,设置Servlet发送的响应数据的MIME类型
创建自己的HttpServlet
下面给出一个实例:
package com.test.servlet; import java.io.IOException; import java.io.PrintWriter; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; public class HelloServlet extends HttpServlet // 第一步: 扩展 HttpServlet 抽象类。 { // 第二步:覆盖doGet()方法 public void doGet(HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException { // 第三步:获取HTTP 请求中的参数信息 String clientName = request.getParameter("clientName"); if (clientName != null) { clientName = new String(clientName.getBytes("ISO-8859-1"), "GB2312"); } else { clientName = "我的朋友"; } // 第四步:生成 HTTP 响应结果。 PrintWriter out; String title = "HelloServlet"; String heading1 = "This is output from HelloServlet by doGet:"; // set content type. response.setContentType("text/html;charset=GB2312"); // write html page. out = response.getWriter(); out.print("<HTML><HEAD><TITLE>" + title + "</TITLE>"); out.print("</HEAD><BODY>"); out.print(heading1); out.println("<h1><P> " + clientName + " : 您好</h1>"); out.print("</BODY></HTML>"); // close out. out.close(); } }
在浏览器地址栏中键入:http://localhost:8080/test/HelloServlet,返回信息为:
This is output from HelloServlet by doGet:
我的朋友 : 您好
键入:http://localhost:8080/test/HelloServlet?clientName=Eric,返回信息为:
This is output from HelloServlet by doGet:
Eric : 您好
Servlet对象何时被创建
- 默认情况下,当Web客户第一次请求访问某个Servlet时,Web容器创建这个Servlet的实例
- 如果设置了<servlet>元素的<load-on-startup>子元素,Servlet容器在启动Web应用时,将按照指定的顺序创建并初始化这个Servlet。如:
<servlet> <servlet-name>HelloServlet</servlet-name> <servlet-class>com.test.HelloServlet</servlet-class> <load-on-startup>2</load-on-startup> </servlet>
ServletContext和Web应用的关系
- 当Servlet容器启动Web应用时,并为每个Web应用创建唯一的ServletContext对象。你可以把ServletContext看成是一个Web应用的服务器端组件的共享内存。在ServletContext中可以存放共享数据,它提供了读取或设置共享数据的方法:-setAttribute(String name, Object object)把一个对象和一个属性名绑定,将这个对象存储在ServletContext中。-getAttribute(String name)根据给定的属性名返回绑定的对象。application就是ServletContext类型的。
Web应用何时被启动
- 当Servlet容器启动时,会启动所有的Web应用
- 通过控制台启动Web应用
创建CounterServlet
package com.test.servlet; import java.io.IOException; import java.io.PrintWriter; import javax.servlet.ServletConfig; import javax.servlet.ServletContext; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; public class CounterServlet extends HttpServlet { private static final String CONTENT_TYPE = "text/html"; public void init(ServletConfig config) throws ServletException // 仅仅执行一次 { super.init(config); System.out.println("init invoked"); } public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { doPost(request, response); } public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { // 获得ServletContext的引用 ServletContext context = getServletContext(); // 从ServletContext读取count属性 Integer count = (Integer) context.getAttribute("count"); // 如果count属性还没有设置, 那么创建count属性,初始值为0 // one and add it to the ServletContext if (count == null) { count = new Integer(0); context.setAttribute("count", new Integer(0)); } response.setContentType(CONTENT_TYPE); PrintWriter out = response.getWriter(); out.println("<html>"); out.println("<head><title>WebCounter</title></head>"); out.println("<body>"); // 输出当前的count属性值 out.println("<p><h1>The current COUNT is : " + count + ".</h1></p>"); out.println("</body></html>"); // 创建新的count对象,其值增1 count = new Integer(count.intValue() + 1); // 将新的count属性存储到ServletContext中 context.setAttribute("count", count); } public void destroy() { } }
测试ConterServlet:
- 通过如下URL访问CounterServlet:http://localhost:8080/test/CounterServlet,当你第一次访问该Servlet时,你会在浏览器上看到count的值为0
- 刷新该页面,你会看到每刷新一次count值增加1,假定最后一次刷新后count值为5
- 再打开一个新的浏览器,访问CounterServlet,此时count值为6
- 重新启动Tomcat服务器,然后再访问CounterServlet,你会看到count值又被初始化为0
Servlet的多线程同步问题(重要)
package com.test.servlet; import java.io.IOException; import java.io.PrintWriter; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; public class HelloServlet extends HttpServlet // 第一步: 扩展 HttpServlet 抽象类。 { String clientName = null; // 第二步:覆盖doGet()方法 public void doGet(HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException { // 第三步:获取HTTP 请求中的参数信息 clientName = request.getParameter("clientName"); if (clientName != null) { clientName = new String(clientName.getBytes("ISO-8859-1"), "GB2312"); } else { clientName = "我的朋友"; } try { Thread.sleep(10000); // 模拟后台复杂处理 } catch(Exception e) { } // 第四步:生成 HTTP 响应结果。 PrintWriter out; String title = "HelloServlet"; String heading1 = "This is output from HelloServlet by doGet:"; // set content type. response.setContentType("text/html;charset=GB2312"); // write html page. out = response.getWriter(); out.print("<HTML><HEAD><TITLE>" + title + "</TITLE>"); out.print("</HEAD><BODY>"); out.print(heading1); out.println("<h1><P> " + clientName + " : 您好</h1>"); out.print("</BODY></HTML>"); // close out. out.close(); } }
在浏览器中同时访问:http://localhost:8080/test/HelloServlet?clientName=李四;http://localhost:8080/test/HelloServlet?clientName=王五,输出结果将被后者覆盖,这是因为Servlet是单例的多线程的,只要把clientName定义成局部变量就可以了。这里解决同步问题最好的方案是:去除实例变量,使用局部变量。
Cookie
- Cookie的英文原意是"点心",它是用户访问Web服务器时,服务器在用户硬盘上存放的信息,好像是服务器送给客户的"点心"。
- 服务器可以根据Cookie来跟踪用户,这对于需要区别用户的场合(如电子商务)特别有用。
- 一个Cookie包含一对Key/Value。下面的代码生成一个Cookie并把它写到用户的硬盘上:Cookie theCookie = new Cookie("cookieName", "cookieValue"); response.addCookie(theCookie);
package com.test.servlet; import java.io.IOException; import javax.servlet.ServletException; import javax.servlet.http.Cookie; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; public class CookieServlet extends HttpServlet { private int count1 = 0; private int count2 = 0; public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { process(request, response); } public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { process(request, response); } public void process(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { Cookie cookie = new Cookie("cookieName" + count1++, "cookieValue" + count2++); cookie.setMaxAge(10); // cookie最大存活时间为10秒钟 response.addCookie(cookie); Cookie[] cookies = request.getCookies(); // 首次刷新浏览器,Console不显示cookie,因为当前request对象不包括cookie信息 if (cookies == null) return; for (int i = 0; i < cookies.length; i++) { System.out.println("Cookie Name :" + cookies[i].getName()); System.out.println("Cookie Value :" + cookies[i].getValue()); } } }
下面是其对应的JSP版本:
<%@ page import="javax.servlet.http.Cookie" %> <html> <head><title>jspCookie.jsp</title></head> <body> <%! int count1 = 0; int count2 = 0; %> <% Cookie cookie = new Cookie("cookieName" + count1++, "cookieValue" + count2++); cookie.setMaxAge(10); response.addCookie(cookie); %> <% Cookie[] cookies = request.getCookies(); if(cookies==null) return; for(int i = 0; i < cookies.length; i++) { %> <p> <b>Cookie name:</b> <%= cookies[i].getName() %> <b>Cookie value:</b> <%= cookies[i].getValue() %> </p> <% } %> </body></html>
对于这个应用而言,Servlet代码更易读一些。
比较Servlet和JSP
- 有许多相似之处,都可以生成动态网页
- JSP的优点是擅长于网页制作,生成动态页面,比较直观。JSP的缺点是不容易跟踪与排错
- Servlet是纯Java语言,擅长于处理流程和业务逻辑。Servlet的缺点是生成动态网页不直观
练习题