HttpServletResponse
1 HttpServletResponse概述
服务器端发送给客户端的数据。
我们在创建Servlet时会覆盖service()⽅法,或doGet()/doPost(),这些⽅法都有两个参数,代表请求的request和代表响应response。
service⽅法中的response的类型是ServletResponse,⽽doGet/doPost⽅法的response的类型是HttpServletResponse,HttpServletResponse是ServletResponse的⼦接⼝,功能和⽅法更加强⼤,所以我们学习HttpServletResponse。
2 response的运⾏流程
3 通过抓包工具抓取Http响应
因为response代表响应,所以我们可以通过该对象分别设置Http响应的响应⾏,响应头和响应体
4 通过response设置响应⾏
响应状态码:服务器告诉客户端浏览器本次请求和响应的⼀个状态。
-
1xx:服务器就收客户端消息,但没有接受完成,等待⼀段时间后,发送1xx多状态码
-
2xx:成功。代表:200
-
3xx:重定向。代表:302(重定向),304(访问缓存)
-
4xx:客户端错误。
代表:
- 404(请求路径没有对应的资源)
- 405:请求⽅式没有对应的doXxx⽅法
-
服务器端错误。代表:500(服务器内部出现异常)
- 设置响应⾏的状态码
setStatus(int sc)
5 通过response设置响应头
常⻅的响应头:
- Content-Type:服务器告诉客户端本次响应体数据格式以及编码格式
- Content-disposition:服务器告诉客户端以什么格式打开响应体数据
- in-line:默认值,在当前⻚⾯内打开
- attachment;filename=xxx:以附件形式打开响应体。⽂件下载
- addHeader(String name, String value)
- addIntHeader(String name, int value)
- addDateHeader(String name, long date)
- setHeader(String name, String value)
- setDateHeader(String name, long date)
- setIntHeader(String name, int value)
其中,add表示添加,⽽set表示设置
6 通过response设置响应体
6.1 响应体设置⽂本
-
PrintWriter getWriter()
获得字符流,通过字符流的write(String s)⽅法可以将字符串设置到response缓冲区中,随后Tomcat会将response缓冲区中的内容组装成Http响应返回给浏览器端。
-
关于设置中⽂的乱码问题
原因:response缓冲区的默认编码是ISO8859-1,此码表中没有中⽂。
可以通过response的setCharacterEncoding(String charset) 设置response的编码(在获取流之前设置)。
-
但我们发现客户端还是不能正常显示⽂字
原因:我们将response缓冲区的编码设置成UTF-8,但浏览器的默认编码是本地系统的编码,因为我们都是中⽂系统,所以客户端浏览器的默认编码是GBK,我们可以⼿动修改浏览器的编码是UTF-8。 -
我们还可以在代码中指定浏览器解析⻚⾯的编码⽅式,
通过response的setContentType(String type)⽅法指定⻚⾯解析时的编码是UTF-8:response.setContentType(“text/html;charset=UTF-8”);上⾯的代码不仅可以指定浏览器解析⻚⾯时的编码,同时也内含 setCharacterEncoding 的功能,所以在实际开发中只要编写response.setContentType(“text/html;charset=UTF-8”);就可以解决⻚⾯输出中⽂乱码问题。
6.2 响应头设置字节
-
ServletOutputStream getOutputStream()
获得字节流,通过该字节流的write(byte[] bytes)可以向response缓冲区中写⼊字节,再由Tomcat服务器将字节内容组成Http响应返回给浏览器。
6.3 案例-完成⽂件下载
⽂件下载的实质就是⽂件拷⻉,将⽂件从服务器端拷⻉到浏览器端。所以⽂件下载需要IO技术将服务器端的⽂件⽤InputStream读取到,再使⽤ServletOutputStream写到response缓冲区中。
// 1.获得download⽂件夹下的a.mp3⽂件的绝对路径
String path = this.getServletContext().getRealPath("/download/a.mp3");
// 2.打开这个⽂件的输⼊流
InputStream is = new BufferedInputStream(new FileInputStream(new
File(path)));
// 3.打开⽂件复制的输出流
ServletOutputStream os = response.getOutputStream();
// 4.复制⽂件
byte[] bs = new byte[1024];
int len = -1;
while ((len = is.read(bs)) > 0) {
os.write(bs, 0, len);
}
// 5.关闭输⼊流
is.close();
// response获得的流不⽤⼿动关闭,会⾃动关闭
上述代码可以将图⽚从服务器端传输到浏览器,但浏览器直接解析图⽚显示在⻚⾯上,⽽不是提
供下载,我们需要设置两个响应头,告知浏览器⽂件的类型和⽂件的打开⽅式。
- 告知浏览器⽂件的类型:response.setContentType(⽂件的MIME类型);
- 告示浏览器⽂件的打开⽅式是下载
response.setContentType(this.getServletContext().getMimeType("a.mp3"));
response.setHeader("Content-Disposition","attachment;filename=a.mp3");
但是,如果下载中⽂⽂件,⻚⾯在下载时会出现中⽂乱码或不能显示⽂件名的情况,原因是不同的浏览器默认对下载⽂件的编码⽅式不同,IE是UTF-8编码⽅式,⽽⽕狐浏览器是Base64编码⽅
式。所⾥这⾥需要解决浏览器兼容性问题,解决浏览器兼容性问题的⾸要任务是要辨别访问者是IE还是⽕狐(其他),通过Http请求体中的⼀个属性可以辨别。
解决乱码的方式如下(了解):
public static String getFileName(String agent, String filename) throws
UnsupportedEncodingException {
if (agent.contains("MSIE")) {
// IE浏览器 %4E
filename = URLEncoder.encode(filename, "utf-8");
filename = filename.replace("+", " ");
} /* else if (agent.contains("Firefox")) {
// ⽕狐浏览器
Base64.Encoder encoder = Base64.getEncoder();
filename = "=?utf-8?B?" + encoder.encode(filename.getBytes("utf-8")) +
"?=";
} */ else {
// 其它浏览器
filename = URLEncoder.encode(filename, "utf-8");
}
return filename;
}
其中agent就是请求头 User-Agent 的值
String filename = "美⼥.jpg";
// 获得浏览器的User-Agent
String agent = request.getHeader("user-agent");
// 解决乱码
String fileName2 = DownLoadUtils.getFileName(agent, filename);
response.setHeader("Content-Disposition","attachment;filename=" +
fileName2);
response.setContentType(this.getServletContext().getMimeType(filename));
7 重定向
重定向:资源跳转的⽅式
代码实现:
//1. 设置状态码为302
response.setStatus(302);
//2.设置响应头location
response.setHeader("location","/day15/responseDemo2");
//简单的重定向⽅法
response.sendRedirect("/day15/responseDemo2");
7.1 重定向和转发
7.1.1 重定向的特点:redirect
- 地址栏发⽣变化
- 重定向可以访问其他站点(服务器)的资源
- 重定向是两次请求。不能使⽤request对象来共享数据
7.1.2 转发的特点:forward
- 转发地址栏路径不变
- 转发只能访问当前服务器下的资源
- 转发是⼀次请求,可以使⽤request对象来共享数据
7.2 路径写法
路径分类
- 相对路径:通过相对路径不可以确定唯⼀资源
- 如:./index.html
- 不以 / 开头,以 . 开头路径
- 规则:找到当前资源和⽬标资源之间的相对位置关系
- ./:当前⽬录
- …/:后退⼀级⽬录
- 绝对路径:通过绝对路径可以确定唯⼀资源
- 如:http://localhost/day15/responseDemo2 /day15/responseDemo2
- 以 / 开头的路径
- 规则:判断定义的路径是给谁⽤的?判断请求将来从哪⼉发出
- 给客户端浏览器使⽤:需要加虚拟⽬录(项⽬的访问路径)
- 建议虚拟⽬录动态获取:request.getContextPath()
- <a>,<form> 重定向…
- 给服务器使⽤:不需要加虚拟⽬录
- 转发路径