Servlet常用对象(上)——HttpServletRequest对象 和 HttpServletResponse对象

Servlet常用对象

请求的方式

  • 地址栏直接输入地址
    • http://IP?或域名:[port]/项目路径/项目中的资源路径?参数名=参数值
  • 超链接
    • <a href=“URL”></a>
  • 表单 form
    • get/post
  • Ajax异步请求
    • 通过 jQuery 中的 ajax(),get(),post(),getJSON()等方法都能发送请求
  • 请求转发
    • 通过服务器内部将请求进行一次转发,可以请求到其他资源
  • 重定向
    • 服务器通过给定一个新资源的地址,响应会客户端后,客户端自动再次发送一个请求到新资源的地址处。

HttpServletRequest 对象

​ 主要作用是用来接收客户端发送过来的请求信息。

常用方法

  • 普通方法

    方法名说明
    getRequestURL()获取请求的完整路径
    getRequestURI()获取部分路径(项目名开始)
    getQueryString()获取请求行中的参数部分
    getMethod()获取请求方式
    getProtocol()获取HTTP的版本号
    getContextPath()获取webapp的名称
  • 获取请求头

    方法名说明
    getHeader(String)根据名称KEY获取单个请求头的内容
    Enumeration<String> getHeaderNames()获取所有请求头的名称枚举集合
  • 获取客户端请求参数

    方法名说明
    getParameter(name)获取指定name属性值的参数
    getParameterValues(String name)获取指定name属性值的所有参数(复选框)
    getParameterNames()获取所有的参数名集合
    getParameterMap()获取参数名和参数值的map对象

请求乱码

  • 乱码场景
    • get请求:Tomcat7及以下版本:乱码;Tomcat8及以上版本:不乱码
    • post请求:Tomcat所有版本都会乱码
  • 乱码原因
    • request默认的解析编码为"ISO-8859-1",不支持中文
  • 解决方式
    • 设置request解析的编码格式支持中文
      • request.setCharacterEncoding(“UTF-8”);
      • 只针对POST请求有效,GET不收任何影响
    • 在接收到乱码数据以后,再通过相应的编码格式还原
      • new String(request.getParameter(“参数名”).getBytes(“ISO-8859-1”), “UTF-8”);
      • GET和POST请求都有效,但是如果本身参数不乱码,使用该方式反而会乱码
    • 一定要在获取参数之前设置编码

请求转发

  • 请求转发是一种服务器行为:当客户端请求到达后,服务器进行转发,此时会将请求对象进行保存
  • 请求转发特点:
    • 地址栏中的 URL 不会改变:得到响应后,服务器端再将响应发送给客户端,从始至终只有一个请求发出
    • 请求转发之间的资源可以共享
  • 请求转发实现方法:
    • request.getRequestDispatcher(“URL”).forward(request, response);
    • 关于请求路径参数URL后面有详细解析

request域对象

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

  • 方法:

    • request.setAttribute(String name, Object obj):设置域对象内容
    • request.getAttribute(String name):获取域对象内容
    • request.removeAttribute(String name):删除域对象内容
  • request 域对象中的数据在一次请求中有效,则经过请求转发,request 域中的数据依然存在

    • 在请求转发的过程中可以通过 request 来传输/共享数据
  • 示例

    • 设置域对象:

      public class Servlet01 extends HttpServlet {
      	private static final long serialVersionUID = 1L;
      
      	protected void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
      		// 设置域对象
      		request.setAttribute("num", 10);
      		request.setAttribute("name", "zhangsan");
      		List<String> list = new ArrayList<>();
      		list.add("aaa");
      		list.add("bbb");
      		list.add("ccc");
      		request.setAttribute("list", list);
      		
      		// 修改作用域
      		request.setAttribute("num", 100);
      		
      		// 移除域对象
      		request.removeAttribute("name");
      		
      		// 请求转发到Servlet02
      		request.getRequestDispatcher("Servlet02").forward(request, response);			
      	}
      }
      
    • 获取域对象:

      public class Servlet08 extends HttpServlet {
      	private static final long serialVersionUID = 1L;
      
      	protected void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
      		// 获取域对象
      		Integer num = (Integer) request.getAttribute("num");
      		System.out.println("num:" + num);
      		String name = (String) request.getAttribute("name");
      		System.out.println("name:" + name);
      		List<String> list = (List<String>) request.getAttribute("list");
      		System.out.println("list:" + list.get(0));				
      	}
      }
      
    • 也可以在jsp动态页面文件中获取域对象

      <h2>欢迎 <%= request.getAttribute("name") %> 登录!</h2>
      

HttpServletResponse 对象

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

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

​ HttpServletResponse 的主要功能用于服务器对客户端的请求进行响应,将 Web 服务器处理后的结果返回给客户端。

常用方法

方法名说明
addHeader(String name, String value)添加指定的键值对到响应头信息中
containsHeader(String name)判断响应的头部是否被设置
encodeURL(String URL)编码指定的URL
sendError(int sc)使用指定状态码发送一个错误到客户端
setHeader(String name, String value)设置指定响应头的值
setStatus(int sc)给当前响应设置状态
setContentType(String ContentType)设置响应的MIME类型
getWriter()获取输出的字符流
getOutputStream()获取输出的字节流

刷新和页面自动跳转

​ 在 response 中一个比较常用的头信息就是刷新的指令,可以完成定时刷新的功能。

// refresh响应头
// response.setHeader("refresh", "2"); // 每隔两秒刷新一次页面
response.setHeader("refresh", "3;url=http://www.baidu.com"); // 几秒钟后跳转到指定页面

数据响应

  • getWriter():获取字符流(只能响应回字符)

    // 获取字符流
    PrintWriter pw = response.getWriter();
    // 输出字符
    pw.write("<h1>Hello World</h1>");
    // 关闭流
    pw.close();
    
  • getOutputStream():获取字节流(能响应一切数据)

    // 获取字节流
    ServletOutputStream sos = response.getOutputStream();
    // 输出字节
    sos.write("<h1>Hello World</h1>".getBytes());
    // 关闭流
    sos.close();
    
  • 响应回的数据到客户端被浏览器解析

  • 注意:两者不能同时使用

响应乱码

  • 乱码原因:
    • 服务器响应的数据也会经过网络传输,服务器端有一种编码方式,在客户端也存在一种编码方式,当两端使用的编码方式不同时则出现乱码
  • getWriter()的字符乱码:
    • 由于getWriter()获取到的字符流,服务器端在进行编码时默认会使用 ISO-8859-1 格式的编码,该编码方式并不支持中文,响应中文必定出乱码
    • 解决方案:
      1. 设置服务器的编码为UTF-8:response.setCharacterEncoding(“UTF-8”);
      2. 设置客户端的编码:response.setHeader(“content-type”, “text/html; charset=UTF-8”);
  • getOutputStream()的字符乱码:
    • 对于getOutputStream()方式获取到的字节流,响应中文时,由于本身就是传输的字节,所以是否会乱码取决于服务端和客户端的编码是否一致
    • 解决方案:
      1. 设置服务器的编码为UTF-8:response.setCharacterEncoding(“UTF-8”);
      2. 设置客户端的编码 :response.setHeader(“content-type”, “text/html; charset=UTF-8”);
  • 总结:设置客户端和服务端的编码一致,且支持中文
    • 同时设置客户端和服务端的编码:response.setContentType(“text/html;charset=UTF-8”);

响应图片

// 设置MIME响应类型
response.setContentType("image/jpeg");
// 获取项目的真实路径(存放在服务器中的路径)
String realPath = request.getServletContext().getRealPath("/");
System.out.println(realPath);
// 图片的路径
String filePath = realPath + "/WEB-INF/jay.jpg";
// 通过路径得到file对象
File file = new File(filePath);
// 判断file是否存在,且为一个标准文件
if (file.exists() && file.isFile()) {
    // 得到字节输出流
    ServletOutputStream out = response.getOutputStream();
    // 得到file对象的输入流
    InputStream in = new FileInputStream(file);
    // 定义byte数组
    byte[] bytes = new byte[1024];
    int len = -1;
    // 遍历
    while((len = in.read(bytes)) != -1) {
        // 输出
        out.write(bytes,0,len);
    }
    // 刷新
    out.flush();
    // 关闭流
    in.close();
    out.close();
}

重定向

  • 重定向是一种服务器指导,客户端的行为
  • 重定向实现方法:
    • request.sendRedirect(“URL”);
  • 请求转发和重定向的区别:
    1. 请求转发是服务器行为,重定向是客户端行为;
    2. 请求转发地址栏不会发生转变,重定向地址栏会改变;
    3. 请求转发只有一次请求,重定向有两次请求;
    4. 请求转发的Request对象可以共享(域对象共享),重定向不可以;
    5. 请求转发定位在站点名后,重定向定位在http://后(重定向可以跨域)

请求路径

  • 请求路径分为两大类:
    1. 相对路径:
      • 相对于当前资源所在的路径(一般前面不需要加"/")
      • 请求转发和重定向的基本路径:http://localhost:8080/站点名/
    2. 绝对路径:
      • 以"http"开头:这种已经跨域(只有重定向可以使用)
      • 以"/"开头:
        • 请求转发:"/"代表的是http://localhost:8080/站点名/
        • 重定向:"/"代表的是http://localhost:8080/
### HttpServletRequestHttpServletResponse 的主要区别 #### 核心概念 `HttpServletRequest` `HttpServletResponse` 是 Java Servlet API 中的核心接口,分别用于表示 HTTP 请求 HTTP 响应。两者的定义用途存在显著差异。 - **HttpServletRequest**: 表示客户端向服务器发送的 HTTP 请求信息[^1]。它封装了请求的所有细节,包括 URL 参数、头部信息、会话数据以及上传文件等内容。 - **HttpServletResponse**: 表示服务器向客户端返回的 HTTP 响应信息[^2]。它允许开发者设置响应的状态码、头部字段以及实际的内容体。 --- #### 生命周期 两者均具有短暂的生命期,在一次完整的 HTTP 请求/响应周期中被创建并销毁: - `HttpServletRequest`: 在每次 HTTP 请求到达时由 Web 容器(如 Tomcat)实例化,并传递至相应的 Servlet 方法(如 `doGet()` 或 `doPost()`)。当请求处理完毕后即被销毁[^3]。 - `HttpServletResponse`: 同样在单次 HTTP 请求期间有效,其生命期与对应的 `HttpServletRequest` 对象一致[^4]。 --- #### 获取方式 这些对象均由 Servlet 容器自动生成并通过方法签名提供给开发人员使用: - 开发者无需手动创建 `HttpServletRequest` 或 `HttpServletResponse` 实例; - 它们作为参数直接传入标准的 Servlet 方法中,例如 `protected void doGet(HttpServletRequest request, HttpServletResponse response)`[^5]。 --- #### 功能特性对比 | 特性 | HttpServletRequest | HttpServletResponse | |---------------------|----------------------------------------------------------------------------------------------------|-----------------------------------------------------------------------------------------------| | **本质** | 封装来自客户端的 HTTP 请求信息 | 构建要返回给客户端的 HTTP 响应 | | **核心功能** | - 提供访问请求参数的方法<br>- 支持读取请求头<br>- 处理会话管理 | - 设置响应状态码<br>- 配置响应头<br>- 输出 HTML/CSS/JSON 等内容 | | **典型操作** | - 调用 `request.getParameter(String name)` 获取表单提交的数据 | - 使用 `response.getWriter().write(String content)` 发送字符串形式的结果 | | | - 利用 `request.getSession(boolean create)` 创建或获取 HttpSession | - 执行 `response.setStatus(int sc)` 修改 HTTP 返回码 | 具体实现方面,虽然表面上我们只接触到这两个接口类型的变量 (`request`, `response`),但实际上底层是由具体的框架类完成支持——比如 Apache Tomcat 中对应的是 `RequestFacade` `ResponseFacade` 类型[^3]。 以下是基于上述理论的一个简单代码片段展示如何联合运用二者: ```java import javax.servlet.ServletException; import javax.servlet.annotation.WebServlet; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException; @WebServlet("/example") public class ExampleServlet extends HttpServlet { @Override protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { String paramValue = request.getParameter("name"); // 从请求中提取名为"name"的参数 if (paramValue != null && !paramValue.isEmpty()) { response.setContentType("text/html;charset=UTF-8"); try (var out = response.getWriter()) { out.println("<html><body>"); out.printf("你好,%s!<br>", paramValue); out.println("</body></html>"); } } else { response.sendError(HttpServletResponse.SC_BAD_REQUEST, "缺少必要参数 'name'"); } } } ``` 此例子清晰展现了怎样利用 `HttpServletRequest` 来解析用户输入的同时借助 `HttpServletResponse` 给予恰当反馈。 --- ### 总结 综上所述,尽管 `HttpServletRequest` `HttpServletResponse` 密切关联于同一个事务流程里,但各自承担完全不同的职责:前者侧重收集前端发起的信息;后者则专注于构建最终呈现给用户的成果集。理解这两者间的异同对于掌握 Java Web 应用程序架构至关重要。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值