一、response響應過程
- 在去發送一個請求時, 會找到tomcat引擎
- 引擎會找到對應的web應用,並且會創建request對象和response對象
- 找到應用后, 會執行應用的web.xml再去根據url-patten的內容創建Servlet對象
- 並且會調用Servlet對象的service方法,並且把創建的request對象和response對象傳入到方法當中
- 拿到response對象后, 自己可以往響應當中寫入一些自己給客戶端的內容,通過response.getwrite().wirte("寫的內容")方法進行寫入
- 寫的內容,是存到一個response緩沖區當中
- 當方法執行結束之后, tomcat就會從response緩沖區當中取出數據
- 取出你的數據同時,它自己還會自動的往里面添加一些服務器相關的信息進去,所以響應給瀏覽器時, 可以看到除了自己寫的內容, 還會有一些服務器相關的信息
流程圖:
二、學習響應
學習通過response設置響應行,響應頭 ,響應體。
1. 設置響應行
response.setState(Int code)
2. 設置響應頭
add 代表添加新的內容
addHeader(String name,String value)
addIntHeader(String name,int value)
addDateHeader(String name,date)
示例:
添加兩個相同的 name
protected void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
response.addHeader("name", "Lucy");
response.addHeader("name", "Mark");
}
set 代表設置已經存在的內容
setHeader(String name,String value)
setIntHeader(String name,int value)
setDateHeader(String name,Date value)
重定向
1. 什么是重定向
- 到服務器當中去找servlet1
- servlet1當中沒有這個資源,告訴你去找servlet2
- 再去發送一個請求到servlet2
2. 狀態碼
- 302
3. 特點
-
要訪問兩次服務器
- 第一次訪問是人為的去訪問
- 第二次是自動的訪問
-
瀏覽器地址欄已經發生變化
4. 設置重定向
- 設置響應碼
- 設置響應頭
protected void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
response.setStatus(302);
response.setHeader("location", "/servlet/firstServlet");
}
5. 封裝的重寫向
- 每次都要寫狀態碼,和location比較麻煩
- 就給封裝了一個方法
response.sendRedirect("/servlet/firstServlet")
6. 定時刷新重定向
response.setHeader("refresh","5;url=http://www.baidu.com")
- 5代表5秒
- url的值為5秒后要去跳轉的地址
3. 設置響應體
1. 通過write方法來寫
response.getWriter().wait("要寫的內容");
-
默認情況下寫的中文內容會亂碼
- 把寫的內容存到緩存區中使用的是ISO8859
- ISO8859不支持中文,所以會亂碼
-
在存之前設置可以設置存放的編碼
response.setCharacterEncoding("UTF-8")
- 主動告知瀏覽器使用的是utf-8編碼
response.setHeader("Content-Type", "text/html;charset=UTF-8");
示例
- 上面代碼只需要寫第二句就行,tom看到設置了為utf-8的編碼,它在存的時候也會使用utf-8的編碼。使用封裝寫法:
response.setContentType("text/html;charset=UTF-8");
2. 通過OutPutStream來寫
FileInputSteam
- read()方法讀取一個字節
- read(byte[] b)
- 一次讀取多個字節,並存放到數組b中
- 上面是一次一滴一滴給你,這種是一次裝一水桶再給你
- 讀取全部的數據
FileInputStream in = new FileInputStream("a.txt");
// 一次性讀取多個字節放到數組當中
byte[] buffer = new byte[5];
// 每次取多少個字節
int len = 0;
while ((len = in.read(buffer)) > 0) {
System.out.println(len);
System.out.println(Arrays.toString(buffer));
System.out.println(new String(buffer, 0, len));
}
in.close();
FileOutputSteam
- write()
- 一次性寫一個字符
- write(buffer)
- 一個性寫多個字符
- write(buffer,0,len)
- 一次性寫指定個數的字符
protected void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//向瀏覽器寫一張圖片
String path = this.getServletContext().getRealPath("code.png");
//加載這張圖片
FileInputStream in = new FileInputStream(path);
//獲取一個輸出流
ServletOutputStream out = response.getOutputStream();
byte[] buffer = new byte[1024];
int len = 0;
while((len = in.read(buffer)) != -1) {
System.out.println(len);
System.out.println(Arrays.toString(buffer));
out.write(buffer, 0, len);
}
}
response注意點
- getWrite()和getOutputSteam不能同時調用
4. 下載功能
需求:把服務器當中的文件直接下載到電腦當中。
下載文件
1.直接使用a標簽來去下載
- 有些內容會瀏覽器自動解析
- 瀏覽器不能解析的文件才會被下載
<body>
<h1>a文件下載</h1>
<a href="/Servlet/download/a.mp4">a.mp4</a><br/>
<a href="/Servlet/download/code.png">code.png</a><br/>
<a href="/Servlet/download/c.rar">c.rar</a>
</body>
2.通過發送Servlet請求來去下載
- 通過發送一個Servlet請求,把文件名發送給服務器
- 發送給服務器后,接收到文件名參數,獲取文件的絕對地址
- 通過流的形式來去寫到瀏覽器
- 還得要告訴文件是什么類型
- 瀏覽器是以MIME的類型來識別類型
- 設置響應的類型
// 瀏覽器是以MIME的類型來識別類型
this.getServletContext().getMimeType(“文件名稱”)
// 設置響應的類型
res.setContentType("MIME類型")
- 設置響應頭,告訴瀏覽器不要去解析,是以附件的形式打開,
res.setHeader("Content-Disposition","attachment;filename="+文件名)
- 步驟
- 1.接收文件名參數
- 2.獲取mime類型
- 3.設置瀏覽器響應類型
- 4.告訴瀏覽器以附件形式下載
- 5.獲取文件的絕對路徑
- 6.讀取文件流
- 7.獲取輸出流
- 8.把內容寫出到輸出流
示例代碼:
// 1.接收文件名參數
String filename = request.getParameter("filename");
// 2.獲取mime類型
String mime = this.getServletContext().getMimeType(filename);
// 3.設置游覽器響應類型
response.setContentType(mime);
// 4.告訴游覽器以附件形式下載
response.setHeader("Content-Disposition", "attachment;filename="+filename);
// 5.獲取文件的絕對路徑
String path = this.getServletContext().getRealPath("download/"+filename);
System.out.println(path);
// 6.讀取文件流
FileInputStream in = new FileInputStream(path);
// 7.獲取輸出流
ServletOutputStream out = response.getOutputStream();
// 8.把內容寫出到輸出流
byte[] buffer = new byte[1024];
int len = 0;
while((len = in.read(buffer)) != -1) {
out.write(buffer, 0, len);
}
in.close();
<h1>servlet文件下載</h1>
<a href="/Servlet/DownloadServlet?filename=a.mp4">a.mp4</a><br/>
<a href="/Servlet/DownloadServlet?filename=code.png">code.png</a><br/>
<a href="/Servlet/DownloadServlet?filename=c.rar">c.rar</a><br/>
<a href="/Servlet/DownloadServlet?filename=二維碼.png">二維碼.png</a><br/>
解決中文名稱亂碼問題:
- 獲取中文參數報錯問題
高版本tomcat中的新特性:就是嚴格按照 RFC 3986規范進行訪問解析,而 RFC 3986規范定義了Url中只允許包含英文字母(a-zA-Z)、數字(0-9)、-_.~4個特殊字符以及所有保留字符(RFC3986中指定了以下字符為保留字符:! * ’ ( ) ; : @ & = + $ , / ? # [ ])
解決方法:
找到Tomcat目錄.../conf/catalina.properties中,找到最后注釋掉的一行
#tomcat.util.http.parser.HttpParser.requestTargetAllow=|
改成
tomcat.util.http.parser.HttpParser.requestTargetAllow=|{}
表示把{}放行
- 把獲取的字符串參數的字節碼獲取,再重新使用utf-8編碼
- 在設置以附件形式打開時, 不同的瀏覽器會對默認的名字進行解碼, 所以根據不同的瀏覽器,要對名稱進行編碼之后,再放入文件名
對文件名進行編碼
- 不同的瀏覽器編碼不一樣
- 要先獲取agent,取出瀏覽器的類型
- 根據不同的瀏覽器類型進行編碼
步驟
- 1.接收文件名稱
- 2.獲取mimeType
- 3.設置瀏覽器響應類型
- 4.先對傳入的參數轉成二進制流,再使用UTF-8進行編碼
- 5.獲取瀏覽器的信息
- 6.判斷是哪一種瀏覽器,根據不同的瀏覽器獲取一個編碼的文件名
- 7.設置以附件形式下載,傳的名稱是編碼過的名稱
- 8.獲取文件的絕對路徑
- 9.讀取文件流
- 10.獲取輸出流
- 11.把文件寫到響應當中
示例代碼:
protected void service(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
/******************* 1.接收請求參數 ***********************************/
// 接收文件名參數
String name = request.getParameter("filename");
System.out.println(name);
/******************* 2.對接收的參數進行編碼處理 **************************/
// 獲取參數 ,默認會對參數進行編碼 ISO8859-1
// 把亂碼轉回二進制位
byte[] bytes = name.getBytes("ISO8859-1");
// 再去使用UTF-8進行編碼
name = new String(bytes, "UTF-8");
System.out.println(name);
/******************* 3.告訴瀏覽器響應的文件的類型 *************************/
// 根據文件名來獲取mime類型
String mimeType = this.getServletContext().getMimeType(name);
// 設置 mimeType
response.setContentType(mimeType);
/******************* 4.告訴瀏覽器以附件的形式下載 *************************/
// 獲取客戶端信息
String agent = request.getHeader("User-Agent");
// 定義一個變量記錄編碼之后的名字
String filenameEncoder = "";
if (agent.contains("MSIE")) {
// IE編碼
filenameEncoder = URLEncoder.encode(name, "utf-8");
filenameEncoder = filenameEncoder.replace("+", " ");
} else if (agent.contains("Firefox")) {
// 火狐編碼
BASE64Encoder base64Encoder = new BASE64Encoder();
filenameEncoder = "=?utf-8?B?" + base64Encoder.encode(name.getBytes("utf-8")) + "?=";
} else {
// 瀏覽器編碼
filenameEncoder = URLEncoder.encode(name, "utf-8");
}
// 告訴瀏覽器是以附件形式來下載 不要解析
response.setHeader("Content-Disposition", "attachment;filename=" + filenameEncoder);
/******************** 5.加載文件響應給瀏覽器 *************************/
// 拼接文件的路徑
String path = this.getServletContext().getRealPath(name);
// 根據path加載文件
FileInputStream in = new FileInputStream(path);
// 獲取響應的輸出
ServletOutputStream out = response.getOutputStream();
byte[] buffer = new byte[1024];
int len = 0;
while ((len = in.read(buffer)) != -1) {
// 寫到響應當中
out.write(buffer, 0, len);
}
// 關閉文件
in.close();
}
解決Eclipse中無法直接使用Base64Encoder的問題
5. 注冊驗證碼功能
為什么要有驗證碼
- 可以防止:惡意批量注冊、惡意破解密碼、刷票、論壇灌水,
- 有效防止某個黑客對某一個特定注冊用戶用特定程序暴力破解方式進行不斷的登陸嘗試。
- 確保當前訪問是來自一個人而非機器
步驟
- 把別人寫好的驗證碼Servlet拿到程序當中
- .編寫靜態頁面
- 編寫判斷驗證碼是否正確Servlet
- 設置響應編碼
- 從ServletContext當中獲取驗證碼
- 獲取傳入的請求參數
- 從servletContext當中取出存放的驗證碼
- 把接收的驗證碼與取出的驗證碼時行比較
- 相同時,顯示成功
- 不同時, 顯示失敗,隔3秒鍾跳轉到輸入驗證碼界面
示例
- CheckCodeServlet.java
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.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
@WebServlet("/CheckCodeServlet")
public class CheckCodeServlet extends HttpServlet {
private static final long serialVersionUID = 1L;
// 集合中保存所有成語
private List<String> words = new ArrayList<String>();
@Override
public void init() throws ServletException {
// 初始化階段,讀取new_words.txt
// web工程中讀取 文件,必須使用絕對磁盤路徑
String path = getServletContext().getRealPath("/WEB-INF/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.setDateHeader("Expires", 0);
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);// 獲得成語
System.out.println(word);
// 定義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);
//把生成的驗證碼存放到全局域對象當中
this.getServletContext().setAttribute("checkCode", 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);
}
}
- RegistServlet.java
import java.io.IOException;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
@WebServlet("/RegistServlet")
public class RegistServlet extends HttpServlet {
private static final long serialVersionUID = 1L;
protected void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//接收請求參數
String code = request.getParameter("code");
System.out.println(code);
byte[] bytes = code.getBytes("ISO8859-1");
code = new String(bytes,"UTF-8");
System.out.println(code);
//取出一開始存放 的word
String word = (String)this.getServletContext().getAttribute("checkCode");
//設置字符集
response.setContentType("text/html;charset=UTF-8");
//兩個做對比
if(code.equals(word)) {
response.getWriter().write("注冊 成功 ");
}else {
response.getWriter().write("驗證碼錯誤 ");
response.setHeader("refresh","3;url=/26-Servlet/code.html");
}
}
}
- code.html
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>Insert title here</title>
<script type="text/javascript">
function change(obj) {
obj.src = "/Servlet/CheckCodeServlet?time=" + new Date().getTime();
}
</script>
</head>
<body>
<form action="/26-Servlet/RegistServlet">
<img src="/26-Servlet/CheckCodeServlet" onclick="change(this)"><br />
<input type="text" placeholder="請輸入驗證碼..." name = "code"><br /><br />
<input type="submit" value="注冊">
</form>
</body>
</html>