百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

基础回顾Servlet系列:request,response,ServletContext

wxin55 2024-10-27 15:57 12 浏览 0 评论

Servlet系列:(HttpServletRequest、HttpServletResponse、ServletContext、ServletConfig)详解

HttpServletRequest

HttpServletRequest 对象:主要作用是用来接收客户端发送过来的请求信息,例如:请求的参数,发送的头信息等都属于客户端发来的信息,service()方法中形参接收的是 HttpServletRequest 接口的实例化对象,表示该对象主要应用在 HTTP 协议上,该对象是由 Tomcat 封装好传递过来。

HttpServletRequest 是 ServletRequest 的子接口,ServletRequest 只有一个子接口,就是HttpServletRequest。既然只有一个子接口为什么不将两个接口合并为一个?

从长远上讲:现在主要用的协议是 HTTP 协议,但以后可能出现更多新的协议。若以后想要支持这种新协议,只需要直接继承 ServletRequest 接口就行了。

在 HttpServletRequest 接口中,定义的方法很多,但都是围绕接收客户端参数的。但是怎么拿到该对象呢?不需要,直接在 Service 方法中由容器传入过来,而我们需要做的就是取出对象中的数据,进行分析、处理。

接收请求

常用方法

  1. getMethod():获取请求的HTTP方法,例如GET、POST等。
  2. getRequestURL():获取请求的URL地址。
  3. getRequestURI():获取请求的URI部分。
  4. getPathInfo():获取请求的路径信息,即URL中“/”之后的部分。
  5. getQueryString():获取请求的查询参数,即URL中“?”后面的部分。
  6. getHeader():获取请求头中的指定字段的值。
  7. getParameter():获取请求参数的值。如果参数不存在,则返回null。
  8. getParameterValues():获取请求参数的所有值。如果参数不存在,则返回null。
  9. getInputStream()和getReader():获取请求的输入流或读取器,用于读取请求的内容。
  10. getSession()和getSession(boolean create):获取与当前请求关联的HttpSession对象。如果当前请求没有关联的HttpSession对象,则可以创建新的HttpSession对象。
  11. getServletPath():获取当前请求的Servlet路径。

示例

// 获取客户端请求的完整URL (从http开始,到?前面结束)
String url = request.getRequestURL().toString(); 
System.out.println("获取客户端请求的完整URL:" + url); 
// 获取客户端请求的部分URL (从站点名开始,到?前面结束) 
String uri = request.getRequestURI(); 
System.out.println("获取客户端请求的部分URL:" + uri); 
// 获取请求行中的参数部分 
String queryString = request.getQueryString(); 
System.out.println("获取请求行中的参数部分:" + queryString); 
// 获取客户端的请求方式 
String method = request.getMethod();
System.out.println("获取客户端的请求方式:" + method); 
// 获取HTTP版本号 
String protocol = request.getProtocol();
System.out.println("获取HTTP版本号:" + protocol); 
// 获取webapp名字 (站点名) 
String webapp = request.getContextPath(); 
System.out.println("获取webapp名字:" + webapp);

请求乱码问题

由于现在的 request 属于接收客户端的参数,所以必然有其默认的语言编码,主要是由于在解析过程中默认使用的编码方式为 ISO-8859-1(此编码不支持中文),所以解析时一定会出现乱码。要想解决这种乱码问题,需要设置 request 中的编码方式,告诉服务器以何种方式来解析数据。或者在接收到乱码数据以后,再通过相应的编码格式还原。

方式一:

这种方式只针对 POST 有效(必须在接收所有的数据之前设定)

request.setCharacterEncoding("UTF-8");

方式二:

new String(request.getParameter(name).getBytes("ISO-8859-1"),"UTF-8")

借助了String 对象的方法,该种方式对任何请求有效,是通用的。

Tomcat8起,以后的GET方式请求是不会出现乱码的。

请求转发

请求转发,是一种服务器的行为,当客户端请求到达后,服务器进行转发,此时会将请求对象进行保存,地址栏中的 URL 地址不会改变,得到响应后,服务器端再将响应发送给客户端,从始至终只有一个请求发出。

实现方式如下,达到多个资源协同响应的效果。

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

request作用域

通过该对象可以在一个请求中传递数据,作用范围:在一次请求中有效,即服务器跳转有效

// 设置域对象内容 
request.setAttribute(String name, String value); 
// 获取域对象内容 
request.getAttribute(String name); 
// 删除域对象内容 
request.removeAttribute(String name);

request 域对象中的数据在一次请求中有效,则经过请求转发,request 域中的数据依然存在,则在请求转发的过程中可以通过 request 来传输/共享数据。

HttpServletResponse

Web服务器收到客户端的http请求,会针对每一次请求,分别创建一个用于代表请求的 request 对象和代表响应的 response 对象。

request 和 response 对象代表请求和响应:获取客户端数据,需要通过 request 对象;向客户端输****出数据,需要通过 response 对象。

HttpServletResponse 的主要功能用于服务器对客户端的请求进行响应,将 Web 服务器处理后的结果返回给客户端。service()方法中形参接收的是 HttpServletResponse 接口的实例化对象,这个对象中封装了向客户端发送数据、发送响应头,发送响应状态码的方法。

响应数据

接收到客户端请求后,可以通过 HttpServletResponse 对象直接进行响应,响应时需要获取输出流。

有两种形式:

  • getWriter() 获取字符流(只能响应回字符)
  • // 字符输出流 PrintWriter writer = response.getWriter(); writer.write("Hello"); writer.write("<h2>Hello</h2>");
  • getOutputStream() 获取字节流(能响应一切数据)
  • // 字节输出流 ServletOutputStream out = response.getOutputStream(); out.write("Hello".getBytes()); out.write("<h2>Hello</h2>".getBytes());

响应回的数据到客户端被浏览器解析。

注意:两者不能同时使用

设置响应类型,默认是字符串

// 设置响应MIME类型 
response.setHeader("content-type","text/html"); // html

响应乱码问题

在响应中,如果我们响应的内容中含有中文,则有可能出现乱码。这是因为服务器响应的数据也会经过网络传输,服务器端有一种编码方式,在客户端也存在一种编码方式,当两端使用的编码方式不同时则出现乱码。

getWriter()的字符乱码

对于 getWriter()获取到的字符流,响应中文必定出乱码,由于服务器端在进行编码时默认会使用 ISO-8859-1 格式的编码,该编码方式并不支持中文。

要解决该种乱码只能在服务器端告知服务器使用一种能够支持中文的编码格式,比如我们通常用的"UTF-8"。

response.setCharacterEncoding("UTF-8");

此时还只完成了一半的工作,要保证数据正确显示,还需要指定客户端的解码方式。

response.setHeader("content-type", "text/html;charset=UTF-8");

两端指定编码后,乱码就解决了。一句话:保证发送端和接收端的编码一致

// 设置服务端的编码 
response.setCharacterEncoding("UTF-8"); 
// 设置客户端的响应类型及编码 
response.setHeader("content-type","text/html;charset=UTF-8"); 
// 得到字符输出流 
PrintWriter writer = response.getWriter(); 
writer.write("<h2>你好</h2>");

以上两端编码的指定也可以使用一句替代,同时指定服务器和客户端

注意:使用set的方法必须在getWriter()方法之前

getOutputStream()字节乱码

对于 getOutputStream()方式获取到的字节流,响应中文时,由于本身就是传输的字节, 所以此时可能出现乱码,也可能正确显示。当服务器端给的字节恰好和客户端使用的编码方式一致时则文本正确显示,否则出现乱码。无论如何我们都应该准确掌握服务器和客户端使用的是那种编码格式,以确保数据正确显示。

指定客户端和服务器使用的编码方式一致。

response.setHeader("content-type","text/html;charset=UTF-8");
// 设置客户端的编码及响应类型 
ServletOutputStream out = response.getOutputStream(); 
response.setHeader("content-type","text/html;charset=UTF-8");
out.write("<h2>你好</h2>".getBytes("UTF-8"));

同样也可以使用一句替代

// 设置客户端与服务端的编码
response.setContentType("text/html;charset=UTF-8");

总结:要想解决响应的乱码,只需要保证使用支持中文的编码格式。并且保证服务器端 和客户端使用相同的编码方式即可

重定向

重定向是一种服务器指导,客户端的行为。客户端发出第一个请求,被服务器接收处理后,服务器会进行响应,在响应的同时,服务器会给客户端一个新的地址(下次请求的地址response.sendRedirect(url);),当客户端接收到响应后,会立刻、马上、自动根据服务器给的新地址发起第二个请求,服务器接收请求并作出响应,重定向完成。

从描述中可以看出重定向当中有两个请求存在,并且属于客户端行为。

// 重定向跳转到index.jsp 
response.sendRedirect("index.jsp");

通过观察浏览器我们发现第一次请求获得的响应码为 302,并且含有一个 location 头信息。并且地址栏最终看到的地址是和第一次请求地址不同的,地址栏已经发生了变化。

请求转发与重定向的区别

请求转发和重定向比较:

两者都可进行跳转,根据实际需求选取即可。

ServletContext

每一个 web 应用都有且仅有一个ServletContext 对象,又称 Application 对象,从名称中可知,该对象是与应用程序相关的。在 WEB 容器启动的时候,会为每一个 WEB 应用程序创建一个对应的ServletContext 对象。

该对象有两大作用:

  • 第一、作为域对象用来共享数据,此时数据在整个应用程序中共享;
  • 第二、该对象中保存了当前应用程序相关信息。例如可以通过 getServerInfo() 方法获取当前服务器信息 ,getRealPath(String path) 获取资源的真实路径等。

ServletContext 对象的获取

获取 ServletContext 对象的途径有很多。比如:

  1. 通过 request 对象获取
  2. @Override public void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { ServletContext servletContext = request.getServletContext(); }
  3. 通过 session 对象获取
  4. @Override public void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { ServletContext servletContext = request.getSession().getServletContext(); }
  5. 通过 servletConfig 对象获取,在 Servlet 标准中提供了 ServletConfig 方法
  6. @Override public void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { ServletContext servletContext = getServletConfig().getServletContext(); }
  7. 直接获取,Servlet 类中提供了直接获取 ServletContext 对象的方法
  8. @Override public void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { ServletContext servletContext = getServletContext(); }

常用方法

// 获取项目存放的真实路径 
String realPath = request.getServletContext().getRealPath("/"); 
// 获取当前服务器的版本信息 
String serverInfo = request.getServletContext().getServerInfo();

ServletContext域对象

ServletContext 也可当做域对象来使用,通过向 ServletContext 中存取数据,可以使得整个应用程序共享某些数据。当然不建议存放过多数据,因为 ServletContext 中的数据一旦存储进去没有手动移除将会一直保存。

// 获取ServletContext对象 
ServletContext servletContext = request.getServletContext(); 
// 设置域对象 
servletContext.setAttribute("name","zhangsan");
// 获取域对象 
String name = (String) servletContext.getAttribute("name"); 
// 移除域对象 
servletContext.removeAttribute("name");

Servlet的三大域对象

  1. request域对象

在一次请求中有效。请求转发有效,重定向失效。

  1. session域对象

在一次会话中有效。请求转发和重定向都有效,session销毁后失效。

  1. servletContext域对象

在整个应用程序中有效。服务器关闭后失效。

ServletConfig

ServletConfig对象是它所对应的Servlet对象的相关配置信息 ,每一个servlet对象都有一个ServletConfig对象和它相对应

  1. getInitParameter(String name) 返回一个初始化变量的值
  2. getInitParameterNames() Enumeration 返回servlet初始化参数的所有名称
  3. getServletContext() 获取ServletContext对象 后面的课程具体讲解ServletContext
  4. getServletName() 获取Servlet的name配置值
    @Override
    public void service(HttpServletRequest request, HttpServletResponse response) throws ServletException,
            IOException {
        ServletConfig servletConfig = getServletConfig();
        servletConfig.getInitParameter("userName"); //
    }

定义

@WebServlet(value = "/context",initParams = {
        @WebInitParam(name="userName",value = "robin")
})

相关推荐

ES6中 Promise的使用场景?(es6promise用法例子)

一、介绍Promise,译为承诺,是异步编程的一种解决方案,比传统的解决方案(回调函数)更加合理和更加强大在以往我们如果处理多层异步操作,我们往往会像下面那样编写我们的代码doSomething(f...

JavaScript 对 Promise 并发的处理方法

Promise对象代表一个未来的值,它有三种状态:pending待定,这是Promise的初始状态,它可能成功,也可能失败,前途未卜fulfilled已完成,这是一种成功的状态,此时可以获取...

Promise的九大方法(promise的实例方法)

1、promise.resolv静态方法Promise.resolve(value)可以认为是newPromise方法的语法糖,比如Promise.resolve(42)可以认为是以下代码的语...

360前端一面~面试题解析(360前端开发面试题)

1.组件库按需加载怎么做的,具体打包配了什么-按需加载实现:借助打包工具(如Webpack的require.context或ES模块动态导入),在使用组件时才引入对应的代码。例如在V...

前端面试-Promise 的 finally 怎么实现的?如何在工作中使用?

Promise的finally方法是一个非常有用的工具,它无论Promise是成功(fulfilled)还是失败(rejected)都会执行,且不改变Promise的最终结果。它的实现原...

最简单手写Promise,30行代码理解Promise核心原理和发布订阅模式

看了全网手写Promise的,大部分对于新手还是比较难理解的,其中几个比较难的点:状态还未改变时通过发布订阅模式去收集事件实例化的时候通过调用构造函数里传出来的方法去修改类里面的状态,这个叫Re...

前端分享-Promise可以中途取消啦(promise可以取消吗)

传统Promise就像一台需要手动组装的设备,每次使用都要重新接线。而Promise.withResolvers的出现,相当于给开发者发了一个智能遥控器,可以随时随地控制异步操作。它解决了三大...

手写 Promise(手写输入法 中文)

前言都2020年了,Promise大家肯定都在用了,但是估计很多人对其原理还是一知半解,今天就让我们一起实现一个符合PromiseA+规范的Promise。附PromiseA+规范地址...

什么是 Promise.allSettled()!新手老手都要会?

Promise.allSettled()方法返回一个在所有给定的promise都已经fulfilled或rejected后的promise,并带有一个对象数组,每个对象表示对应的pr...

前端面试-关于Promise解析与高频面试题示范

Promise是啥,直接上图:Promise就是处理异步函数的API,它可以包裹一个异步函数,在异步函数完成时抛出完成状态,让代码结束远古时无限回掉的窘境。配合async/await语法糖,可...

宇宙厂:为什么前端离不开 Promise.withResolvers() ?

大家好,很高兴又见面了,我是"高级前端进阶",由我带着大家一起关注前端前沿、深入前端底层技术,大家一起进步,也欢迎大家关注、点赞、收藏、转发。1.为什么需要Promise.with...

Promise 新增了一个超实用的 API!

在JavaScript的世界里,Promise一直是处理异步操作的神器。而现在,随着ES2025的发布,Promise又迎来了一个超实用的新成员——Promise.try()!这个新方法简...

一次搞懂 Promise 异步处理(promise 异步顺序执行)

PromisePromise就像这个词的表面意识一样,表示一种承诺、许诺,会在后面给出一个结果,成功或者失败。现在已经成为了主流的异步编程的操作方式,写进了标准里面。状态Promise有且仅有...

Promise 核心机制详解(promise机制的实现原理)

一、Promise的核心状态机Promise本质上是一个状态机,其行为由内部状态严格管控。每个Promise实例在创建时处于Pending(等待)状态,此时异步操作尚未完成。当异步操作成功...

javascript——Promise(js实现promise)

1.PromiseES6开始支持,Promise对象用于一个异步操作的最终完成(包括成功和失败)及结果值的表示。简单说就是处理异步请求的。之所以叫Promise,就是我承诺,如果成功则怎么处理,失败怎...

取消回复欢迎 发表评论: