1.Response的运行流程
2.通过抓包工具抓取Http响应
通过response设置响应行:
设置响应行的状态码:setStatus(int sc)
设置响应头:
addHeader(String name,String value) , addIntHeader(String name,Intvalue) , addDataHeader(String name,Long value)
setHeader(String name,String value) , setIntHeader(String name,Int value) , setDateHeader(String name,Long value)
其中,add表示添加(添加多个可以全部显示),而set表示设置(只能显示一个)
设置响应体一:向response写入字符流:PrintWriter getWriter()方法可以获取字符流,通过write()方法把字符串写到response缓冲区,之后tom内核把response缓冲区的内容组装成http响应返回给客户端。
中文乱码问题:原因:response缓冲区的编码格式是ISO8859-1,此码表没有中文,可通过response的setCharacterEncoding()设置response的编码。但还会乱码,因为浏览器默认编码是GBK,需要通过response.setHeader("Content-Type", "text/html;charset=utf-8")告知客户端使用何种编码格式进行解析。
还有一种更加简洁的方法:response.setContentType("text/html;charset=utf-8"),这个方法可以同时指定response和浏览器解析的编码,所以在实际开发中使用这个方法。
字节和字符:字节(byte)是计算存储容量的单位,字符是计算机中使用的文字和符号,字符串。
设置响应体二:向Response写入字节流:ServletOutputStream getOutputStream()方法获得字节流,通过字节流的write(byte[] bytes)可以将图片视频文本等文件写入response缓冲区,再由tomcat内核组装成http响应返回给浏览器。
注意:response.getOutputStream()和response.getWriter()只能同时存在一个。
写入图片示例:
ServletOutputStream out = response.getOutputStream();
//获得服务器上的图
String path = this.getServletContext().getRealPath("download/a.jpg");
InputStream in=new FileInputStream(path);//fileInputStream 是InputStream 的实现类
int len=0;
byte[] buffer=new byte[1024];
while((len=in.read(buffer))>0) {
out.write(buffer, 0, len);
}
out.close();
in.close();
案例:文件下载(中文乱码处理):
响应头的filename属性要传给客户端,客户端需要有对应的解码方式,否则会乱码
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>Insert title here</title>
</head>
<body>
<a href="/Test002/download?filename=美.jpg">美.jpg</a><br>
<a href="/Test002/download?filename=a.flv">a.flv</a><br>
<a href="/Test002/download?filename=a.mp3">a.mp3</a><br>
</body>
</html>
package com.xing.download;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.URLEncoder;
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 sun.misc.BASE64Encoder;
@WebServlet("/download")
public class download extends HttpServlet {
private static final long serialVersionUID = 1L;
protected void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
String filename = request.getParameter("filename");//此处显示中文正常,浏览器能识别
//filename属性是下载的文件名称,但传入中文会乱码,解决方法:
//filename属性设置方式一:
// String fname=URLEncoder.encode(filename,"UTF-8");
// 告知浏览器文件打开方式是下载,filename属性是下载的文件名称,但此处传入的是中文,会乱码。解决:filename="+fname+";"+"filename*=utf-8''"+fname
// response.setHeader("Content-Disposition", "attachment;filename="+fname+";"+"filename*='utf-8'"+fname);
//filename属性设置方式二:针对不同浏览器设置不同解码方式
//获得请求头中的User-Agent
String agent = request.getHeader("User-Agent");
//根据不同浏览器进行不同的编码
String filenameEncoder = "";
if (agent.contains("MSIE")) {
// IE浏览器
filenameEncoder = URLEncoder.encode(filename, "utf-8");
filenameEncoder = filenameEncoder.replace("+", " ");
} else if (agent.contains("Firefox")) {
// 火狐浏览器
BASE64Encoder base64Encoder = new BASE64Encoder();
filenameEncoder = "=?utf-8?B?"
+ base64Encoder.encode(filename.getBytes("utf-8")) + "?=";
} else {
// 其它浏览器
filenameEncoder = URLEncoder.encode(filename, "utf-8");
}
//告知浏览器文件打开方式是下载,filename属性是下载的文件名称,传入中文浏览器不能解析
response.setHeader("Content-Disposition", "attachment;filename="+filenameEncoder);
ServletOutputStream out = response.getOutputStream();
String path = this.getServletContext().getRealPath("download/" + filename);
InputStream in = new FileInputStream(path);
byte[] buffer = new byte[1024];
int len;
while ((len = in.read(buffer)) > 0) {
out.write(buffer, 0, len);
}
in.close();
}
protected void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
doGet(request, response);
}
}
乱码心得:使用getWriter().write向response写入字符流乱码,使用response.setContentType("text/html;charset=utf-8")解决(一般出现在页面中文乱码);使用setHeader()把中文写入响应头,浏览器不能解析,乱码(下载名称中文乱码),需要针对不同浏览器设置解析格式。
3.重定向流程图(sendRedirect())
response.sendRedirect("login.html");
案例:验证码:
package cn.itcast.estore.web.servlet;
import java.awt.Color;
import java.awt.Font;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.image.BufferedImage;
import java.io.BufferedReader;
import java.io.FileReader;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
import java.util.Random;
import javax.imageio.ImageIO;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
/**
* 验证码生成程序
*
*
*
*/
public class CheckImgServlet extends HttpServlet {
// 集合中保存所有成语
private List<String> words = new ArrayList<String>();
@Override
public void init() throws ServletException {
// 初始化阶段,读取new_words.txt
// web工程中读取 文件,必须使用绝对磁盘路径
String path = getServletContext().getRealPath("/WEB-INF/new_words.txt");
try {
BufferedReader reader = new BufferedReader(new FileReader(path));
String line;
while ((line = reader.readLine()) != null) {
words.add(line);
}
reader.close();
} catch (IOException e) {
e.printStackTrace();
}
}
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
// 禁止缓存
// response.setHeader("Cache-Control", "no-cache");
// response.setHeader("Pragma", "no-cache");
// response.setDateHeader("Expires", -1);
int width = 120;
int height = 30;
// 步骤一 绘制一张内存中图片
BufferedImage bufferedImage = new BufferedImage(width, height,
BufferedImage.TYPE_INT_RGB);
// 步骤二 图片绘制背景颜色 ---通过绘图对象
Graphics graphics = bufferedImage.getGraphics();// 得到画图对象 --- 画笔
// 绘制任何图形之前 都必须指定一个颜色
graphics.setColor(getRandColor(200, 250));
graphics.fillRect(0, 0, width, height);
// 步骤三 绘制边框
graphics.setColor(Color.WHITE);
graphics.drawRect(0, 0, width - 1, height - 1);
// 步骤四 四个随机数字
Graphics2D graphics2d = (Graphics2D) graphics;
// 设置输出字体
graphics2d.setFont(new Font("宋体", Font.BOLD, 18));
Random random = new Random();// 生成随机数
int index = random.nextInt(words.size());
String word = words.get(index);// 获得成语
// 定义x坐标
int x = 10;
for (int i = 0; i < word.length(); i++) {
// 随机颜色
graphics2d.setColor(new Color(20 + random.nextInt(110), 20 + random
.nextInt(110), 20 + random.nextInt(110)));
// 旋转 -30 --- 30度
int jiaodu = random.nextInt(60) - 30;
// 换算弧度
double theta = jiaodu * Math.PI / 180;
// 获得字母数字
char c = word.charAt(i);
// 将c 输出到图片
graphics2d.rotate(theta, x, 20);
graphics2d.drawString(String.valueOf(c), x, 20);
graphics2d.rotate(-theta, x, 20);
x += 30;
}
// 将验证码内容保存session
request.getSession().setAttribute("checkcode_session", word);
// 步骤五 绘制干扰线
graphics.setColor(getRandColor(160, 200));
int x1;
int x2;
int y1;
int y2;
for (int i = 0; i < 30; i++) {
x1 = random.nextInt(width);
x2 = random.nextInt(12);
y1 = random.nextInt(height);
y2 = random.nextInt(12);
graphics.drawLine(x1, y1, x1 + x2, x2 + y2);
}
// 将上面图片输出到浏览器 ImageIO
graphics.dispose();// 释放资源
//将图片写到response.getOutputStream()中
ImageIO.write(bufferedImage, "jpg", response.getOutputStream());
}
public void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
doGet(request, response);
}
/**
* 取其某一范围的color
*
* @param fc
* int 范围参数1
* @param bc
* int 范围参数2
* @return Color
*/
private Color getRandColor(int fc, int bc) {
// 取其随机颜色
Random random = new Random();
if (fc > 255) {
fc = 255;
}
if (bc > 255) {
bc = 255;
}
int r = fc + random.nextInt(bc - fc);
int g = fc + random.nextInt(bc - fc);
int b = fc + random.nextInt(bc - fc);
return new Color(r, g, b);
}
}
一唱一和
一呼百应
一干二净
一举两得
一落千丈
一模一样
一暴十寒
一日千里
一五一十
一心一意
两面三刀
三长两短
三番五次
三三两两
三头六臂
三心二意
三言两语
四分五裂
四面八方
四通八达
四平八稳
五光十色
五湖四海
五花八门
五颜六色
六神无主
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>Insert title here</title>
<script type="text/javascript">
function changeCode(obj){
obj.src="/Test002/code?time="+new Date().getTime();
}
</script>
</head>
<body>
<div style="width=100px;height=100px;">
验证码:<input type="text" name="username"><img onclick="changeCode(this)" src="/Test002/code"><br/>
</div>
</body>
</html>
补充:服务器端WEB-INF/classes文件夹相当于项目工程的src文件夹,类是编译过的。读取WEB-INF/classes里的文件,可通过ServletContext对象或者类加载器。
@Override
protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//servletContext读取WEB-INF下的web.xml
ServletContext servletContext = this.getServletContext();
String driver = servletContext.getInitParameter("driver");
System.out.println(driver);
//类加载器读取读取WEB-INF下的txt
String path = getNum.class.getClassLoader().getResource("../new_words.txt").getPath();//路径从classes里的com开始,注意txt文件位置
System.out.println(path);
}
读取包下的txt路径改为:getResource("com/xing/getWeb/new_words.txt")