响应消息:服务器端发送给客户端的数据
数据格式
1.响应行
组成:协议/版本 响应状态码 状态码的描述
响应状态码:服务器告诉客户端浏览器本次请求和响应的状态,状态码由三位数字组成,大致分为五类
状态码的分类:
1xx:服务器接收客户端消息但没有接收完成,等待一段时间后发送1xx状态码
2xx:代表这次请求响应成功,典型:200
3xx:代表重定向,典型:302(重定向),304(访问缓存)
4xx:代表客户端错误,典型:404(请求路径没有对应的资源),405(请求方式没有对应的doXxx方法)
5xx:代表服务器端错误,典型:500(服务器内部出现异常)
2.响应头
格式: 头名称:值
常见的响应头:
Content-Type:服务器告诉客户端浏览器本次响应体的数据格式和编码格式
Content-Disposition:服务器告诉客户端以什么格式打开响应体数据
取值:
in-line:默认值,在当前页面内打开
attachment;filename=xxx:以附件形式打开响应体,文件下载
3.响应空行
4.响应体:传输的数据
响应消息的字符串格式
HTTP/1.1 200 OK
Content-Type: text/html;charset=UTF-8
Content-Length: 101
Date: Wed, 06 Jun 2018 07:08:42 GMT
<html>
<head>
<title>$Title$</title>
</head>
<body>
hello , response
</body>
</html>
Response对象
功能:设置响应消息
1.设置响应行
格式:http/1.1 200 ok
设置状态码:setStatus(int sc)
2.设置响应头
setHeader(String name, String value)
3.设置响应体
步骤:
1.获取输出流
字符输出流:PrintWriter getWriter()
字节输出流:ServletOutputStream getOutputStream()
2.使用流对象,将数据输出到客户端浏览器
案例
1.完成重定向(重定向:资源跳转的一种方式)
重定向的特点:
1.浏览器地址发生变化
2.重定向可以访问其他站点(服务器)的资源
3.重定向是两次请求,不能使用request域共享数据
转发的特点:
1.浏览器地址不发生变化
2.只能转发到当前服务器内部资源
3.转发是一次请求,可以使用request域共享数据
路径写法:
路径的分类:
相对路径:通过相对路径不可确定唯一的资源,不以/斜杠开头,以点.开头
如:./index.html(./代表当前路径,可以省略,不写就默认为./ 如:index.html)
规则:确定当前资源和目标资源之间的相对位置关系
绝对路径:通过绝对路径可以确定唯一的资源,以/斜杠开头
如:http://localhost/day24/responseDemo2,可以简写为/day24/responseDemo2
规则:判断定义的路径是给谁用的(判断请求时从哪发出的);虚拟目录:项目的访问路径
给客户端浏览器使用:需要加上虚拟目录(如:重定向);建议虚拟目录动态获取
给服务器使用:不需要加虚拟目录(如:转发)
动态获取虚拟目录:request.getContextPath()
例:
package com.fengqi.web.servlet;
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;
/**
* @Author: NieFengQi
* @Date: 2019/7/28 19:25
* @Decription:重定向到resposeDemo02
*/
@WebServlet("/responseDemo01")
public class ResponseDemo01 extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
System.out.println("demo01");
/*//设置状态码和location的头信息
response.setStatus(302);
response.setHeader("location","/day24/responseDemo02");*/
//使用sendRedirect方法完成重定向
response.sendRedirect("/day24/responseDemo02");
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
this.doPost(request,response);
}
}
package com.fengqi.web.servlet;
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;
/**
* @Author: NieFengQi
* @Date: 2019/7/28 19:25
* @Decription:
*/
@WebServlet("/responseDemo02")
public class ResponseDemo02 extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
System.out.println("demo02");
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
this.doPost(request,response);
}
}
2.服务器输出字符数据到浏览器
步骤:
1.获取字符输出流对象
2.输出数据
中文乱码问题
产生乱码的原因:response.getWriter()是获取流对象,这个流是tomcat创建的,默认编码为ISO-8859-1,而浏览器的默认编码是操作系统的编码(GBK),所以产生乱码。若是自己创建的流对象默认编码和操作系统一致。
解决:在获取流之前response.setContentType(“text/html;charset=UTF-8”);
这段代码会告诉浏览器,服务器发送消息体数据的编码,并建议浏览器使用该编码
例:
package com.fengqi.web.servlet;
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;
import java.io.PrintWriter;
/**
* @Author: NieFengQi
* @Date: 2019/7/29 17:08
* @Decription:
*/
@WebServlet("/responseDemo03")
public class ResponseDemo03 extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//获取流之前设置流的编码为utf-8,流的默认编码是ISO-8859-1
response.setContentType("text/html;charset=UTF-8");
//获取字符输出流
PrintWriter pw = response.getWriter();
//输出数据
pw.write("response-->响应");
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
this.doPost(request,response);
}
}
3.服务器输出字节数据到浏览器
步骤:
1.获取字节输出流对象
2.输出数据
中文乱码问题如上解决,将编码设置为GBK
例:
package com.fengqi.web.servlet;
import javax.servlet.ServletException;
import javax.servlet.ServletOutputStream;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
/**
* @Author: NieFengQi
* @Date: 2019/7/29 18:49
* @Decription:
*/
@WebServlet("/responseDemo04")
public class ResponseDemo04 extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//告诉浏览器响应体的编码
response.setContentType("text/html;charset=utf-8");
//获取字节输出流
ServletOutputStream sos = response.getOutputStream();
//输出数据,使用getBytes将数据转换成字节数组
sos.write("你好response".getBytes());
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
this.doPost(request,response);
}
}
4.验证码案例
本质:图片
目的:防止恶意注册
例:
逻辑代码:
package com.fengqi.web.servlet;
import javax.imageio.ImageIO;
import javax.servlet.ServletException;
import javax.servlet.ServletOutputStream;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.awt.*;
import java.awt.image.BufferedImage;
import java.io.IOException;
import java.util.Random;
/**
* @Author: NieFengQi
* @Date: 2019/7/29 19:07
* @Decription:
*/
@WebServlet("/checkCodeServlet")
public class CheckCodeServlet extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//创建对象,该对象在内存中代表一张图片(验证码对象)
int width = 100;
int height = 50;
BufferedImage image = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);
//美化验证码
Graphics g = image.getGraphics();//画笔对象
g.setColor(Color.PINK);//设置画笔颜色
g.fillRect(0,0,width,height);//填充背景色
g.setColor(Color.BLUE);
g.drawRect(0,0,width-1,height-1);//画边框
//随机生成四个码
String str = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz1234567890";
Random random = new Random();
for (int i = 1; i <= 4; i++) {
int index = random.nextInt(str.length());
char ch = str.charAt(index);
g.drawString(ch+"",width/5*i,height/2);
}
//设置干扰线
g.setColor(Color.GREEN);
for (int i = 0; i < 10; i++) {
int x1 = random.nextInt(width);
int x2 = random.nextInt(width);
int y1 = random.nextInt(height);
int y2 = random.nextInt(height);
g.drawLine(x1,y1,x2,y2);
}
//将验证码输出到页面展示
response.setContentType("text/html;charset=utf-8");
ServletOutputStream sos = response.getOutputStream();
ImageIO.write(image,"jpg",sos);
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
this.doPost(request,response);
}
}
html页面
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
<script>
window.onload = function () {
//通过id获取图片对象
var img = document.getElementById('checkCode');
//给图片对象绑定单击事件
img.onclick = function () {
//获取当前时间的毫秒值
var date = new Date().getTime();
//加上?date可以让浏览不使用缓存图片
img.src = "/response/checkCodeServlet?"+date;
}
//获取超链接对象
var link = document.getElementById("change");
//给超链接绑定单击事件
link.onclick = function () {
//将超链接的href设置为当前页面
link.href = "#";
//获取获取当前时间的毫秒值
var date = new Date().getTime();
//更改图片色src属性,加上?date可以让浏览不使用缓存图片
img.src = "/response/checkCodeServlet?"+date;
}
}
</script>
</head>
<body>
<img id="checkCode" src="/response/checkCodeServlet">
<a id="change" href="#">看不清,换一张?</a>
</body>
</html>