1. 程式人生 > 資訊 >資料顯示:中國在智慧手機使用者方面領先於其他所有國家,現擁有超 9.5 億人

資料顯示:中國在智慧手機使用者方面領先於其他所有國家,現擁有超 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) 下載檔案

  1. 要獲取下載檔案的路徑

  2. 下載的檔名是什麼?

  3. 設定下個辦法讓瀏覽器支援下載我們需要的東西

  4. 獲取下載檔案的輸入流

  5. 建立緩衝區

  6. 獲取OutputStream物件

  7. 將FileOutputStream寫入到buffer緩衝區

  8. 使用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);
    }
}