JSP中Servlet的Request与Response的用法与区别
简单来说,每一次Web交互的核心,都围绕着两个关键对象展开。当浏览器向服务器发起一个HTTP请求时,服务器会立刻创建一对孪生兄弟:一个是代表入站信息的Request对象,另一个是代表出站信息的Response对象。想要读懂客户端说了什么?找Request就行。想要给客户端回点什么?那就得靠Response了。理解了它们,基本上就摸清了Servlet数据流转的主干道。
一、Request:解读客户端意图
Request对象封装了一次HTTP请求的全部细节,从请求行、请求头到实体内容,它提供了一系列方法,让我们能够精准地获取客户端的各类信息。
1. 作为容器
setAttribute(String name,Object o) // 将数据作为request对象的一个属性存放到request对象中 getAttribute(String name) // 获取request对象的name属性的属性值 removeAttribute(String name) // 移除request对象的name属性 getAttributeNames() // 获取request对象的所有属性名
这组方法让Request对象临时扮演了一个“数据中转站”的角色。你可以在一次请求的生命周期内,在不同处理环节之间传递数据,非常方便。
2. 获得客户端信息
getRequestURL() // 返回客户端发出请求时的完整URL。 getRequestURI() // 返回请求行中的资源名部分。 getRemoteAddr() // 返回发出请求的客户机的IP地址。 getRemoteHost() // 返回发出请求的客户机的完整主机名。 getRemotePort() // 返回客户机所使用的网络端口号。 getLocalAddr() // 返回WEB服务器的IP地址。 getLocalName() // 返回WEB服务器的主机名。 getQueryString() // 返回请求行中的参数部分。 getPathInfo() // 返回请求URL中的额外路径信息。额外路径信息是请求URL中的位于Servlet的路径之后和查询参数之前的内容,它以“/”开头。
这部分方法堪称“信息搜集器”,从客户端的网络地址到请求的具体目标,都能一览无余。
3. 获得客户端请求头
getHeader(string name) getHeaders(String name) getHeaderNames()
HTTP请求头里藏着不少“秘密”,比如浏览器类型、支持的语言、会话标识等。这些方法就是解读这些“秘密”的钥匙。
4. 获得客户端请求参数(客户端提交的数据)
getParameter(String)方法(常用) getParameterValues(String name)方法(常用) getParameterNames()方法(不常用) getParameterMap()方法(编写框架时常用)
这可能是日常开发中最常用的一组方法了。无论是表单提交的字段,还是URL后追加的参数,都靠它们来提取。
5. request接收表单提交中文参数乱码问题解决
说到处理请求参数,中文乱码是个绕不开的“坑”。问题根源很明确:服务器和客户端沟通的编码不一致。但解决方式,根据请求方法的不同,得分成两种情况看:
(1)GET方式提交表单:
有意思的是,对于GET请求,即使在服务器端调用request.setCharacterEncoding("UTF-8"),也常常无效。这是因为GET参数附在URL中,Tomcat等服务器默认会用ISO8859-1去解码。这就好比对方用普通话喊话,你却用方言去听,自然听不懂。
解决办法:
第一种是“手动转码”:先按服务器默认的ISO8859-1拿到原始字节,再用正确的编码(如UTF-8)重新组装成字符串。
String name = request.getParameter("name");
name = new String(name.getBytes("ISO8859-1"), "UTF-8");
第二种更一劳永逸,直接在服务器的server.xml配置文件中,为连接器(Connector)加上URIEncoding="UTF-8"属性,从根本上统一URL的编码。
(2)POST方式提交表单:
POST请求的解决方式就直观多了。因为其参数在请求体中,只需在获取参数之前,明确告诉Request对象该用什么编码去解读即可。
request.setCharacterEncoding("UTF-8");
6. Request对象实现请求转发
request.getRequestDispatcher("/test.jsp").forward(request, response);
request.getRequestDispatcher("/test.jsp").include(request, response);
这两个方法是服务器内部跳转的利器。forward()是“彻底接管”,转向新地址后,之前的响应内容会被清空。而include()则是“内容融合”,将目标页面的结果包含进来,与当前页面的内容合并输出。区分好这一点,在组装页面时就不会混淆。
二、Response:构建服务器回音
如果说Request是倾听者,那么Response就是发言者。响应的状态码、响应头和实体内容,都由它来主导设置。
1. 向客户端(浏览器)发送数据的相关方法
getOutputStream() // 获取字节输出流,常用于输出二进制数据(如图片、文件)。 getWriter() // 获取字符输出流,常用于输出文本内容(如HTML、JSON)。
需要警惕的是,这两个流在同一响应中不能同时使用,否则会抛出异常。根据你要输出的内容类型,二选其一即可。
2. 向客户端(浏览器)发送响应头的相关方法
addDateHeader() addHeader() addIntHeader() containsHeader() setDateHeader() setHeader() setIntHeader()
响应头控制着浏览器如何理解接下来的内容。比如,下面这行代码就告诉浏览器:“我接下来发的是UTF-8编码的HTML文本,请按这个方式渲染。”
response.setHeader("content-type", "text/html;charset=UTF-8");
其中,setHeader()会覆盖同名的头,而addHeader()则可以添加多个同名的头值。
3. 向客户端(浏览器)发送响应状态码的相关方法
最常用的就是setStatus()方法,用来设置如200(成功)、404(未找到)、500(服务器错误)等HTTP状态码。
实践示例:文件下载
理论说再多,不如看一个经典案例。下面这段代码清晰地展示了如何组合运用Response的各种方法来实现文件下载功能:
private void downloadFileByOutputStream(HttpServletResponse response)
throws FileNotFoundException, IOException {
//1.获取要下载的文件的绝对路径
String realPath = this.getServletContext().getRealPath("/download/1.JPG");
//2.获取要下载的文件名
String fileName = realPath.substring(realPath.lastIndexOf("\\")+1);
//3.设置content-disposition响应头,控制浏览器以下载形式打开文件
response.setHeader("content-disposition", "attachment;filename="+fileName);
//4.获取要下载的文件输入流
InputStream in = new FileInputStream(realPath);
int len = 0;
//5.创建数据缓冲区
byte[] buffer = new byte[1024];
//6.通过response对象获取OutputStream流
OutputStream out = response.getOutputStream();
//7.将FileInputStream流写入到buffer缓冲区
while ((len = in.read(buffer)) > 0) {
//8.使用OutputStream将缓冲区的数据输出到客户端浏览器
out.write(buffer,0,len);
}
in.close();
}
关键在于第3步:通过设置content-disposition响应头为attachment
核心区别总结
聊了这么多具体用法,二者的核心区别其实已经呼之欲出了。从角色定位上:**Request是信息的“索取者”与“承载者”**,它身上附带了客户端的所有请求信息,包括参数、IP、Cookie、表单数据乃至上传的文件。你的工作是通过它提供的各种get方法,准确“解读”客户端的意图。
而**Response是数据的“发送者”与“构造者”**。你的任务是通过它设置状态、头部信息,并利用其自带的输出流,将处理好的页面、数据或文件“组装”并“递送”回客户端。
一句话概括:一个主内(接收解析),一个主外(响应输出)。深刻理解这对对象的协作机制,是掌握Ja va Web开发基础的关键一步。
希望以上的梳理能为大家厘清思路。如果在实践中遇到具体问题,欢迎留言探讨。感谢阅读!