Servlet请求和响应

作者: 程序熊大 | 来源:发表于2017-06-17 22:17 被阅读2396次

在Java Web中Servlet、请求和响应是最基本的三个对象,在Web容器的管理下,这三者能够完成基本的HTTP请求处理。

Servlet的作用是为客户提供服务。servlet的角色是接受一个客户的请求,再返回一个响应。请求可能非常简单,例如:给我提供一个欢迎页面;也可能非常复杂,例如:为当前的购物车结账,这个请求会带一些客户端传来的参数,servlet需要知道自己如何使用请求中的参数,还需要知道该返回什么样的响应。

一、Servlet

1. Servlet受容器管理

Java Web服务器处理用户请求的基本过程:用户在客户端点击一个链接,浏览器会向Web应用服务器发送一个URL请求,该URL会指向一个servlet;Web容器看出这个请求指向某个servlet A,就会创建两个对象(HttpServletRequest和HttpServletResponse),并分配或创建一个线程,调用servlet A对应的service方法(上述请求和响应对象作为参数);service根据HTTP请求区分出客户端发来的是GET还是POST请求,并调用对应的doGet()或doPost()方法;在doGet()或doPost()方法中进行业务逻辑的处理,处理完成后的结果通过响应对象返回写回给客户端。

2. Servlet的生命周期

在容器启动时,XXXServlet在JVM的管理下被实例化为一个对象,这时候它还不是servlet,需要在容器的管理下调用init()方法进行初始化,获得ServletConfig和ServletContext对象的引用后,才称为一个真正的Servlet。

Servlet的一生.png
  1. init()
  • 何时调用?servlet实例创建后,并在servlet能为客户请求提供服务之前,容器需要通过init方法初始化servlet。
  • 作用?初始化servlet实例,使之获得web容器的相关信息
  • 是否会被覆盖?有可能
  1. service()
  • 何时调用?当一个客户请求到来时,容器会创建(或从线程池分配一个线程),并调用servlet的service方法
  • 作用?这个方法会确定HTTP方法(GET or POST),并调用对应的servlet方法——doGet或doPost;
  • 是否会被覆盖?可以,但是不会
  1. doGet()或doPost()
  • 何时调用?
  • 作用?具体的业务逻辑
  • 是否会被覆盖?至少要覆盖其中之一

关键点:每个请求都在一个单独的线程中运行!

3. Servlet的继承体系

  • servlet接口:javax.servlet.Servlet,表示所有的Servlet都有这5个方法,其中init、service和destroy三个方法和servlet的生命周期有关
  • GenericServlet:javax.servlet.GenericServlet,这是一个抽象类,它实现了开发者需要的大部分基本servlet方法,大多数servlet的“servlet行为”都来自这个类;
  • HttpServlet:javax.servlet.http.HttpServlet,这也是一个抽象类,它实现了自己的service()方法,处理servlet的HTTP特性(service方法不仅仅只处理HTTP请求)。
  • MyTestServlet:这是开发者自己编写的处理类,一般只需要实现doGet()和doPost()方法。


    servlet的继承体系.png

二、请求和响应

1. ServletRequest的继承体系

HttpServletRequest的API与HTTP有关,例如:Cookie、首部(Header)和会话(Session)等;


HttpServletRequest的继承体系.png

2. ServletResponse的继承体系

ServletResponse(响应)也是类似,用于帮助servlet给客户端返回处理结果,而HttpServletResponse增加了HTTP相关的内容(例如:错误、cookie和首部)等API。

HttpServletResponse的继承体系.png

HttpServletRequest和HttpServletResponse这些都是servlet规范里指定的接口,而web容器开发商(例如tomcat)会负责实现这些接口,例如:HttpServletResponseWrapper和ApplicationHttpResponse等,作为开发者,我们只需要知道,在处理doGet()和doPost方法时,容器会给这个方法传HttpServletRequest和HttpServletResponse两个参数。

3. GET和POST的区别

  • POST方法有请求体
  • GET方法的查询参数直接跟在URL后面,不够安全;
  • GET请求可以建立书签,POST请求则不能
  • GET请求是幂等的,POST请求不是(GET请求仅仅用于查询一些数据,POST请求则用于在服务器上更新数据),在业务上会遇到既需要POST请求,又需要保证请求幂等的情况(例如库存扣减),这种情况需要我们出具对应的实现方案。参见:

4. HTTP请求的API

  • getHeader(),可以获取首部信息,例如request.getHeader("User-Agent")可以获取客户端的平台和浏览器信息。
  • getIntHeader(),如果首部信息中的“key/value”对中的value是int类型的,可以使用这个方法直接获取值而不需要显式类型转换
  • getCookies(),可以获取与请求相关的cookie
  • getSession(),可以获取与请求相关的会话
  • getMethod(),可以获取http方法
  • getInputStream(),可以获取请求的输入流
  • getParameter(String name),可以获取HTTP请求的请求参数,对于GET请求,可以获取查询字符串中的数据、对于POST请求,可以获取请求体中的数据
  • getRemotePort(),获取客户端的端口号
  • getServerPort(),获取服务端接受请求的端口号(请求一开始发送服务端的哪个端口?)
  • getLocalPort(),获取服务端处理请求的端口号(请求最后是发送到服务端的哪个端口?)

5. HTTP响应的API

大多数情况下,使用响应只是为了向客户发回数据。会对响应调用两个方法:setContentType()和getWriter()。在此之后,可以将HTML或其他内容写入到流。不过,你也可以使用响应设置首部、发送错误或增加Cookie。

  • setContentType(),设置响应返回的MIME类型
  • getOutputStream(),获取HTTP输出字节流
  • getWriter(),获取HTTP输出字符流
  • addCookie(Cookie cookie),给响应首部中增加cookie对象,注意这里不是增加“key/value”对
  • addHeader(),在响应首部中添加一个“key/value”对
  • setHeader(),在响应首部中设置一个“key/value”对;和addHeader()的区别是,如果响应首部中已经有对应的key存在,setHeader()会覆盖现有的值,而addHeder()会新增一个“key/value”对,使用时需要注意
  • encodeRedirectURL(),对包含session ID的URL进行编码。使用场景:在浏览器不支持使用cookie跟踪会话时,可以使用URL重写(即将URL重定向到另一个URL,而这个URL的后面会带上session id传给客户端,这个URL在返回给客户端之前需要经过编码)
    /**
     * Encodes the specified URL for use in the <code>sendRedirect</code> method
     * or, if encoding is not needed, returns the URL unchanged. The
     * implementation of this method includes the logic to determine whether the
     * session ID needs to be encoded in the URL. Because the rules for making
     * this determination can differ from those used to decide whether to encode
     * a normal link, this method is separated from the <code>encodeURL</code>
     * method.
     * <p>
     * All URLs sent to the <code>HttpServletResponse.sendRedirect</code> method
     * should be run through this method. Otherwise, URL rewriting cannot be
     * used with browsers which do not support cookies.
     *
     * @param url
     *            the url to be encoded.
     * @return the encoded URL if encoding is needed; the unchanged URL
     *         otherwise.
     * @see #sendRedirect
     * @see #encodeUrl
     */
    public String encodeRedirectURL(String url);
  • sendError(),给客户端返回错误的响应(错误码、错误信息),在该方法被调用之后,就意味着响应已经被返回给客户端,也就不能再调用response的任何方法,否则会报IllegalStateException。
    /**
     * Sends an error response to the client using the specified status code and
     * clears the output buffer. The server defaults to creating the response to
     * look like an HTML-formatted server error page containing the specified
     * message, setting the content type to "text/html", leaving cookies and
     * other headers unmodified. If an error-page declaration has been made for
     * the web application corresponding to the status code passed in, it will
     * be served back in preference to the suggested msg parameter.
     * <p>
     * If the response has already been committed, this method throws an
     * IllegalStateException. After using this method, the response should be
     * considered to be committed and should not be written to.
     *
     * @param sc
     *            the error status code
     * @param msg
     *            the descriptive message
     * @exception IOException
     *                If an input or output exception occurs
     * @exception IllegalStateException
     *                If the response was committed
     */
    public void sendError(int sc, String msg) throws IOException;
  • setStatus(),设置响应的状态码
  • sendRedirect(),用于URL重定向,告诉客户端去访问另一个URL来完成需求,如果location没有以“/”开头,则它是相对路径,容器会负责将这个相对路径转化成该web应用的绝对路径;否则容器会把它当做绝对路径处理。
    /**
     * Sends a temporary redirect response to the client using the specified
     * redirect location URL. This method can accept relative URLs; the servlet
     * container must convert the relative URL to an absolute URL before sending
     * the response to the client. If the location is relative without a leading
     * '/' the container interprets it as relative to the current request URI.
     * If the location is relative with a leading '/' the container interprets
     * it as relative to the servlet container root.
     * <p>
     * If the response has already been committed, this method throws an
     * IllegalStateException. After using this method, the response should be
     * considered to be committed and should not be written to.
     *
     * @param location
     *            the redirect location URL
     * @exception IOException
     *                If an input or output exception occurs
     * @exception IllegalStateException
     *                If the response was committed or if a partial URL is given
     *                and cannot be converted into a valid URL
     */
    public void sendRedirect(String location) throws IOException;

6. 重定向和请求派发

  • 重定向是让浏览器访问新的URL完成工作,用户会在浏览器地址栏看到新的URL;
  • 请求派发是服务端的工作,是当前servlet委托另外的servlet完成请求,并给客户端发回响应,用户的浏览器地址栏的URL没有改变;

三、总结

  1. 理解servlet的生命周期
  2. 理解servlet处理客户请求的线程模型
  3. 理解servlet的继承体系
  4. 理解servlertRequest的继承体系
  5. 理解servlertResponse的继承体系
  6. 熟悉servlet可以从HTTP请求获取哪些内容,在实际应用中有什么作用
  7. 熟悉servlert可以使用HTTP响应给客户端返回什么内容,在实际应用中的场景
  8. 理解重定向和请求派发的区别

相关文章

网友评论

  • c46797261327:写的很有条理,很有用,能顺便问一下这是什么编辑器吗?在哪里可以下载?:+1:
    程序熊大: @Bryant帅 IDEA
    c46797261327:@杜琪 :smile: 我是想问一下,写的Java代码用的什么编辑器。。。能告诉一下吗?
    程序熊大:@Bryant帅 谢谢肯定。简书自带的markdown编辑器,无需下载:wink:

本文标题:Servlet请求和响应

本文链接:https://www.haomeiwen.com/subject/ojmjqxtx.html