一、Http传输响应的的消息结构
(回顾)请求的消息:
1、请求行
2、请求头
3、请求空行
4、请求体
响应的消息:
1、响应行
组成:协议/版本 响应状态码 状态码描述
响应状态码:是服务器告诉浏览器本次响应的状态
1、状态码:是3位数字
2、状态码分类:
1.1XX :服务器接收客户端消息,没有接收完成,等待一段时间后,发送1XX状态码
2.2XX:成功 200
3.3XX:重定向(302)重定向 (304)访问缓存
4.4xx:客户端错误 (404)请求路径没有相应的资源 (405) 请求方式中没有doXXX方法(post/get)
5.5XX:服务器端错误 (500)服务器内部出现异常
2、响应头
1、格式:头名称:值
常见响应头:
Content-Type:服务器告诉客户端响应体以什么数据格式和编码格式
Content-disposition:服务器告诉客户端以什么格式打开响应体数据
in-line:默认值,在页面中打开
attachment;filename=xxx:以附件的形式打开响应体,文件下载
3、响应空行
4、响应体:传输数据(html页面。。。)
二、Response对象
功能:设置响应消息
1、设置响应行
格式:http/1.1 200 ok
设置状态码:setStatus(int sc)
2、设置响应头 setHeader(String name,String val);
3、设置响应体:
使用步骤:
1、获取字符输出流:PrintWriter getWriter()
2、获取字节输出流:ServletOutputStream getOutputStream()
demo:完成重定向
//方式1
reponse.setStatus(302);//重定向的状态码为302
reponse.setHeader("location",“/day15/responseDemo2”);//设置响应头
//方式2
reponse.sendRedirect("/day15/responseDemo2")
重定向和转发的区别
转发
1、地址栏不改变
2、只能访问当前服务器下的资源
3、转发是一次请求,可以使用request对象来共享资源
重定向:
1、地址栏改变
2、可以访问其他服务器资源
3、重定向是两次请求,不能用request对象来共享资源
路径的写法
1、路径的分类:
相对路径:不可以确定唯一资源
如:./:当前目录
../:上一级目录
2、绝对路径:通过绝对路径可以确定唯一资源
如:http://localhost/day15/responseDemo2 /day15/responseDemo2
规则:判断定义的路径是给谁用的,请求从哪里发出
给客户端浏览器使用:request.getContextPath()
给服务器使用:不需要加虚拟目录
服务器输出字符数据到浏览器
1、获取字符流
2、 输出数据
* 注意:
* 乱码问题:
1. PrintWriter pw = response.getWriter();获取的流的默认编码是ISO-8859-1
2. 设置该流的默认编码
3. 告诉浏览器响应体使用的编码//简单的形式,设置编码,是在获取流之前设置
response.setContentType("text/html;charset=utf-8");3. 服务器输出字节数据到浏览器
* 步骤:
1. 获取字节输出流
2. 输出数据4. 验证码
1. 本质:图片
2. 目的:防止恶意表单注册
ServletContext对象
1. 概念:代表整个web应用,可以和程序的容器(服务器)来通信
2. 获取:
1. 通过request对象获取
request.getServletContext();
2. 通过HttpServlet获取
this.getServletContext();
3. 功能:
1. 获取MIME类型:
* MIME类型:在互联网通信过程中定义的一种文件数据类型
* 格式: 大类型/小类型 text/html image/jpeg* 获取:String getMimeType(String file)
2. 域对象:共享数据
1. setAttribute(String name,Object value)
2. getAttribute(String name)
3. removeAttribute(String name)* ServletContext对象范围:所有用户所有请求的数据
3. 获取文件的真实(服务器)路径
1. 方法:String getRealPath(String path)
String b = context.getRealPath("/b.txt");//web目录下资源访问
System.out.println(b);
String c = context.getRealPath("/WEB-INF/c.txt");//WEB-INF目录下的资源访问
System.out.println(c);
String a = context.getRealPath("/WEB-INF/classes/a.txt");//src目录下的资源访问
System.out.println(a);
案例
案例:
* 文件下载需求: 1. 页面显示超链接 2. 点击超链接后弹出下载提示框 3. 完成图片文件下载* 分析: 1. 超链接指向的资源如果能够被浏览器解析,则在浏览器中展示,如果不能解析,则弹出下载提示框。不满足需求 2. 任何资源都必须弹出下载提示框 3. 使用响应头设置资源的打开方式: * content-disposition:attachment;filename=xxx* 步骤: 1. 定义页面,编辑超链接href属性,指向Servlet,传递资源名称filename 2. 定义Servlet 1. 获取文件名称 2. 使用字节输入流加载文件进内存 3. 指定response的响应头: content-disposition:attachment;filename=xxx 4. 将数据写出到response输出流
问题:
* 中文文件问题
* 解决思路:
1. 获取客户端使用的浏览器版本信息
2. 根据不同的版本信息,设置filename的编码方式不同
DownLoadUtils 工具类,解决浏览器下载文件名不能识别中文字符等问题
package com.woniuxy.utils;
import sun.misc.BASE64Encoder;
import java.io.UnsupportedEncodingException;
import java.net.URLEncoder;
public class DownLoadUtils {
public static String getFileName(String agent, String filename) throws UnsupportedEncodingException {
if (agent.contains("MSIE")) {
// IE浏览器
filename = URLEncoder.encode(filename, "utf-8");
filename = filename.replace("+", " ");
} else if (agent.contains("Firefox")) {
// 火狐浏览器
BASE64Encoder base64Encoder = new BASE64Encoder();
filename = "=?utf-8?B?" + base64Encoder.encode(filename.getBytes("utf-8")) + "?=";
} else {
// 其它浏览器
filename = URLEncoder.encode(filename, "utf-8");
}
return filename;
}
}