資料顯示:中國在智慧手機使用者方面領先於其他所有國家,現擁有超 9.5 億人
Servlet
一、Servlet簡介
- Servlet就是sun公司開發動態web的一門技術
- sun在這些API中提供一個介面叫做:Servlet,如果你想開發Servlet程式,只需完成兩個小步驟:
- 編寫一個類,實現Servlrt介面
- 把開發好的java類部署到web伺服器中
把實現了Servlet介面的java程式叫做,Servlet
二、HelloServlet
Servlet介面Sun公司有兩個預設的實現類:HttpServlet,GenericServlet
1、構建一個普通的Maven專案,刪掉裡面所有的src資料夾,以後學習就在這個專案裡面建立Moudel,這個空的工程就是Maven主工程
2、關於Maven父子工程的理解:
父專案中的pom.xml會有
<modules>
<module>servlet-01</module>
</modules>
子專案中的pom.xml會有
<parent> <artifactId>javaweb-02-servlet</artifactId> <groupId>com.jhoves</groupId> <version>1.0-SNAPSHOT</version> </parent>
父專案中的java子專案可以直接使用
3、Maven環境優化
(1)修改web.xml為最新的
<?xml version="1.0" encoding="UTF-8"?> <web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd" version="3.1" metadata-complete="true"> </web-app>
(2)將maven的結構搭建完整
4、編寫一個Servlet程式
(1)編寫一個普通類
(2)實現Servlet介面,這裡直接繼承HttpServlet
package com.jhoves.servlet;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;
public class HelloServlet extends HttpServlet {
//由於get或者post只是請求實現的不同方式,可以相互呼叫,以為業務邏輯都一樣
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// ServletOutputStream outputStream = resp.getOutputStream();
PrintWriter writer = resp.getWriter();//響應流
writer.print("Hello,Serlvet");
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
super.doPost(req, resp);
}
}
(3)編寫Servlet的對映
為什麼需要對映:我們寫的是Java程式,但是要通過瀏覽器訪問,而瀏覽器需要連線web伺服器,所以我們需要在web服務中註冊我們寫的Servlet,還需給它一個瀏覽器能夠訪問的路徑
**接著在web.xml中配置:**
```xml
<!--註冊Servlet-->
<servlet>
<servlet-name>hello</servlet-name>
<servlet-class>com.jhoves.servlet.HelloServlet</servlet-class>
</servlet>
<!--Servlet的請求路徑-->
<servlet-mapping>
<servlet-name>hello</servlet-name>
<url-pattern>/hello</url-pattern>
</servlet-mapping>
```
(4)配置Tomcat
解決方法
注意:配置專案釋出的路徑就可以了
(5)啟動測試
三、Servlet原理
Servlet是由Web伺服器呼叫,web伺服器在收到瀏覽器請求之後,會:
四、mapping問題
1、一個Servle可以指定一個對映路徑
<servlet-mapping>
<servlet-name>hello</servlet-name>
<url-pattern>/hello</url-pattern>
</servlet-mapping>
2、一個Servle可以指定多個對映路徑
<servlet-mapping>
<servlet-name>hello</servlet-name>
<url-pattern>/hello</url-pattern>
</servlet-mapping>
<servlet-mapping>
<servlet-name>hello</servlet-name>
<url-pattern>/hello1</url-pattern>
</servlet-mapping>
<servlet-mapping>
<servlet-name>hello</servlet-name>
<url-pattern>/hello2</url-pattern>
</servlet-mapping>
<servlet-mapping>
<servlet-name>hello</servlet-name>
<url-pattern>/hello3</url-pattern>
</servlet-mapping>
3、一個Servle可以指定通用對映路徑
<servlet-mapping>
<servlet-name>hello</servlet-name>
<url-pattern>/hello/*</url-pattern>
</servlet-mapping>
4、預設請求路徑
<servlet-mapping>
<servlet-name>hello</servlet-name>
<url-pattern>/*</url-pattern>
</servlet-mapping>
5、指定一些字尾或者字首等等
<!--可以自定義字尾實現請求對映
注意點,*前面不能加專案對映的路徑
hello/sajdlkajda.qingjian
-->
<servlet-mapping>
<servlet-name>hello</servlet-name>
<url-pattern>*.qingjiang</url-pattern>
</servlet-mapping>
6、優先順序問題
指定了固有的對映路徑優先順序最高,如果找不到就會走預設的處理請求;
五、ServletContext
web容器在啟動的時候,它會為每個web程式都建立一個對應的ServletContext物件,它代表了當前的web應用;
1、共享資料
我在這Servlet中儲存的資料,可以在另一個Servlet中拿到;
設定的類
public class HelloServlet extends HttpServlet{
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// this.getInitParameter() 初始化引數
// this.getServletConfig() Servlet配置
// this.getServletContext() Servlet上下文
ServletContext Context = this.getServletContext();
String username = "蕪湖";//資料
Context.setAttribute("username",username);//將一個數據儲存在了ServletContext中,名字為username,值為username
}
}
獲取的類
public class GetServlet extends HttpServlet{
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
ServletContext context = this.getServletContext();
String username = (String) context.getAttribute("username");
resp.setCharacterEncoding("UTF-8");
resp.setContentType("text/html");
resp.getWriter().print("名字"+username);
}
}
web.xml配置
<servlet>
<servlet-name>getc</servlet-name>
<servlet-class>com.jhoves.servlet.GetServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>getc</servlet-name>
<url-pattern>/getc</url-pattern>
</servlet-mapping>
測試訪問結果:
直接../getc:
先../hello後再../getc結果:
2、獲取初始化引數
xml配置引數
<!--配置一些web應用初始化引數-->
<context-param>
<param-name>url</param-name>
<param-value>jdbc:mysql://localhost:3306/mybatis</param-value>
</context-param>
獲取資料
public class ServletDemo03 extends HttpServlet{
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
ServletContext context = this.getServletContext();
String url = context.getInitParameter("url");
resp.getWriter().print(url);
}
}
xml註冊
<servlet>
<servlet-name>gp</servlet-name>
<servlet-class>com.jhoves.servlet.ServletDemo03</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>gp</servlet-name>
<url-pattern>/gp</url-pattern>
</servlet-mapping>
結果
3、請求轉發
轉發路徑不變,重定向路徑改變
public class ServletDemo04 extends HttpServlet{
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
ServletContext context = this.getServletContext();
System.out.println("進入了ServletDemo04");
RequestDispatcher requestDispatcher = context.getRequestDispatcher("/gp");//轉發的請求路徑
requestDispatcher.forward(req,resp);//呼叫forward實現請求轉發
}
}
4、讀取資原始檔
Properties
- 在java目錄下新建Properties
- 在resources目錄下新建Properties
發現:都被打包到同一個路徑下:classes,我們俗稱這個路徑為classpath;
思路:需要一個檔案流;
username=root
password=123456
public class ServletDemo05 extends HttpServlet{
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
InputStream is = this.getServletContext().getResourceAsStream("/WEB-INF/classes/db.properties");
Properties prop = new Properties();
prop.load(is);
String user = prop.getProperty("username");
String pwd = prop.getProperty("password");
resp.getWriter().print(user+":"+pwd);
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req,resp);
}
}
訪問測試即可;
六、HttpServletResponse
web伺服器接收到客戶端的http請求,針對這個請求,分別建立一個代表請求的HttpServletRequest物件,代表響應的一個HttpServletResponse;
- 如果要獲取客戶端請求過來的引數:找HttpServletRequest
- 如果要給客戶端響應一些資訊:找HttpServletResponse;
1、簡單分類
負責向瀏覽器傳送資料的方法
ServletOutputStream getOutputStream() throws IOException;
PrintWriter getWriter() throws IOException;
負責向瀏覽器傳送響應頭的方法
void setCharacterEncoding(String var1);
void setContentLength(int var1);
void setContentLengthLong(long var1);
void setContentType(String var1);
void setDateHeader(String var1, long var2);
void addDateHeader(String var1, long var2);
void setHeader(String var1, String var2);
void addHeader(String var1, String var2);
void setIntHeader(String var1, int var2);
void addIntHeader(String var1, int var2);
響應的狀態碼
int SC_CONTINUE = 100;
int SC_SWITCHING_PROTOCOLS = 101;
int SC_OK = 200;
int SC_CREATED = 201;
int SC_ACCEPTED = 202;
int SC_NON_AUTHORITATIVE_INFORMATION = 203;
int SC_NO_CONTENT = 204;
int SC_RESET_CONTENT = 205;
int SC_PARTIAL_CONTENT = 206;
int SC_MULTIPLE_CHOICES = 300;
int SC_MOVED_PERMANENTLY = 301;
int SC_MOVED_TEMPORARILY = 302;
int SC_FOUND = 302;
int SC_SEE_OTHER = 303;
int SC_NOT_MODIFIED = 304;
int SC_USE_PROXY = 305;
int SC_TEMPORARY_REDIRECT = 307;
int SC_BAD_REQUEST = 400;
int SC_UNAUTHORIZED = 401;
int SC_PAYMENT_REQUIRED = 402;
int SC_FORBIDDEN = 403;
int SC_NOT_FOUND = 404;
int SC_METHOD_NOT_ALLOWED = 405;
int SC_NOT_ACCEPTABLE = 406;
int SC_PROXY_AUTHENTICATION_REQUIRED = 407;
int SC_REQUEST_TIMEOUT = 408;
int SC_CONFLICT = 409;
int SC_GONE = 410;
int SC_LENGTH_REQUIRED = 411;
int SC_PRECONDITION_FAILED = 412;
int SC_REQUEST_ENTITY_TOO_LARGE = 413;
int SC_REQUEST_URI_TOO_LONG = 414;
int SC_UNSUPPORTED_MEDIA_TYPE = 415;
int SC_REQUESTED_RANGE_NOT_SATISFIABLE = 416;
int SC_EXPECTATION_FAILED = 417;
int SC_INTERNAL_SERVER_ERROR = 500;
int SC_NOT_IMPLEMENTED = 501;
int SC_BAD_GATEWAY = 502;
int SC_SERVICE_UNAVAILABLE = 503;
int SC_GATEWAY_TIMEOUT = 504;
int SC_HTTP_VERSION_NOT_SUPPORTED = 505;
2、下載檔案
(1) 向瀏覽器輸出訊息
(2) 下載檔案
-
要獲取下載檔案的路徑
-
下載的檔名是什麼?
-
設定下個辦法讓瀏覽器支援下載我們需要的東西
-
獲取下載檔案的輸入流
-
建立緩衝區
-
獲取OutputStream物件
-
將FileOutputStream寫入到buffer緩衝區
-
使用OutputStream物件將緩衝區中的資料輸出到客戶端
案例:
public class FileServlet extends HttpServlet{
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// 1. 要獲取下載檔案的路徑
String realPath = "F:\\JavaIDEAcode\\JavaWeb\\javaweb-02-servlet\\response\\target\\classes\\黃一.png";
System.out.println("下載檔案的路徑"+realPath);
// 2. 下載的檔名是什麼?
String filename = realPath.substring(realPath.lastIndexOf("\\") + 1);
// 3. 設定下個辦法讓瀏覽器支援(Content-Disposition)下載我們需要的東西,中文檔名URLEncoder.encode編碼,否則有可能亂碼
resp.setHeader("Content-Disposition","attachment;filename="+ URLEncoder.encode(filename,"UTF-8"));
// 4. 獲取下載檔案的輸入流
FileInputStream in = new FileInputStream(realPath);
// 5. 建立緩衝區
int len = 0;
byte[] buffer = new byte[1024];
// 6. 獲取OutputStream物件
ServletOutputStream out = resp.getOutputStream();
// 7. 將FileOutputStream寫入到buffer緩衝區,使用OutputStream物件將緩衝區中的資料輸出到客戶端
while ((len=in.read(buffer))>0){
out.write(buffer,0,len);
}
in.close();
out.close();
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
3、驗證碼功能
驗證怎麼來的?
- 前端實現
- 後端實現,需要用到 java 的圖片類,生成一個圖片
public class ImageServlet extends HttpServlet{
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//如何讓瀏覽器五秒自動重新整理一次
resp.setHeader("refresh","3");
//在記憶體中建立一個圖片
BufferedImage image = new BufferedImage(80, 20, BufferedImage.TYPE_INT_RGB);
//得到圖片
Graphics2D g = (Graphics2D)image.getGraphics();//筆
//設定圖片的背景顏色
g.setColor(Color.white);
g.fillRect(0,0,80,20);
//給圖片寫資料
g.setColor(Color.blue);
g.setFont(new Font(null,Font.BOLD,20));
g.drawString(makeNum(),0,20);
//告訴瀏覽器這個請求用圖片的方式開啟
resp.setContentType("image/jpeg");
//網站存在快取,不讓瀏覽器快取
resp.setDateHeader("expires",-1);
resp.setHeader("Cache-Control","no-cache");
resp.setHeader("Pragma","no-cache");
//把圖片寫給瀏覽器
ImageIO.write(image, "jpg",resp.getOutputStream());
}
//生成隨機數
private String makeNum(){
Random random = new Random();
String num = random.nextInt(9999999) + "";
StringBuffer sb = new StringBuffer();
for (int i = 0; i < 7 - num.length(); i++) {
sb.append("0");
}
num=sb.toString()+num;
return num;
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
4、實現重定向(掌握)
一個web資源收到客戶端請求後,它會通知客戶端去訪問另一個web資源,這個過程就叫重定向
常見場景:
- 使用者登入
主要方法:
void sendRedirect(String var1) throws IOException;
案例:
public class RedirectServlet extends HttpServlet{
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
/*
原理:
resp.setHeader("Location","/r/img");
resp.setStatus(302)
*/
resp.sendRedirect("/r/img");
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
面試題
請說出重定向和轉發的區別?
相同點
- 頁面都會實現跳轉
不同點
- 請求轉發的時候,URL不會產生變化 307
- 重定向的時候,URL位址列會發生變化 302
七、HttpServletRequest
HttpServletRequest代表客戶端的請求,使用者通過Http協議訪問呢伺服器,Http請求中的所有資訊會被封裝到HttpServletRequest,通過這個HttpServletRequest的方法,獲得客戶端的所有資訊。
獲取前端傳遞的引數並且請求轉發
public class LoginServlet extends HttpServlet{
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
req.setCharacterEncoding("UTF-8");
String usesname = req.getParameter("usesname");
String password = req.getParameter("password");
String[] hobbies = req.getParameterValues("hobbies");
System.out.println("===============================================");
//後臺接受中文亂碼問題
System.out.println(usesname);
System.out.println(password);
System.out.println(Arrays.toString(hobbies));
System.out.println("===============================================");
//通過請求轉發
//resp.setCharacterEncoding("UTF-8");
//這裡的/代表當前的web應用,重定向才要/r
req.getRequestDispatcher(req.getContextPath()+"/success.jsp").forward(req,resp);
//重定向
//resp.sendRedirect("/r/success.jsp");
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}