重拾Java Web應用的基礎體系結構

1、背景

  • Spring生態的強大與完善,使得大多數的Java程序員,在剛剛接觸Java Web應用開發時,每每依賴於SSM、SpringBoot等各類高級框架。
  • Java Web的基礎的體系結構是什麼?究竟是怎麼運做的?這些高級的框架與基礎的體系結構之間是什麼關係?
  • 只有真正理清了這些底層基礎的結構,才能徹底理解高級框架的設計原理,在使用框架開發項目時作到事半功倍。
  • 本文旨在暫時拋開這些高級框架,重走Java Web底層之路。

2、Web應用

  • Web應用的基礎模型
  1. 用戶經過Web瀏覽器向某個Web應用發出一個請求
  2. Web服務器獲得請求後,處理並查找資源,並向Web瀏覽器返回一個響應結果。
    在這裏插入圖片描述

2.1 HTML

  • Web服務器須要對客戶端的請求提供響應內容,Web瀏覽器須要將這些響應內容呈現給用戶。二者須要造成統一的對話語言,Web應用中造成的共同的語言被稱爲HTML(HyperText Markup Language:超文本標記語言)。html

  • HTML包含數十個標記,數千種標記屬性,能過這些標記定義了網頁內容的含義和結構;正是無數個具備相互連接的HTML網頁構成了咱們如今的互聯網世界。java

標記 描述
html 定義HTML文檔的邊界
head 定義HTML文檔頭部的邊界
body 定義HTML文檔本體的邊界
tile 定義HTML文檔本體的標題
form 定義一個表單
a 定義一個超連接
... ...
<!-- HTML基礎 -->
<html>
  <head>
    <title>歡迎</title>
  </head>
  <body>
   <a href="https://www.baidu.com">百度</a>
  </body>
</html>

2.2 HTTP

  • Web瀏覽器與Web服務器之間相互通訊一樣須要創建統一的協議:Web應用的標準協議被稱爲HTTP(Hyper Text Transfer Protocol)超文本傳輸協議。
  • HTTP協議規定了超文本傳輸所要遵照的規則:HTTP請求可發起經常使用的GET或者POST請求;HTTP的響應中能夠包含HTML內容。
    在這裏插入圖片描述
curl https://www.baidu.com -v
<!-- 經過HTTP向URL地址www.baidu.com發起 GET請求-->
> GET / HTTP/1.1
> User-Agent: curl/7.29.0
> Host: www.baidu.com
> Accept: */*
<!-- Web服務器HTTP響應-->
< HTTP/1.1 200 OK
< Accept-Ranges: bytes
< Cache-Control: private, no-cache, no-store, proxy-revalidate, no-transform
< Connection: keep-alive
< Content-Length: 2443
< Content-Type: text/html
< Date: Tue, 25 Aug 2020 04:04:01 GMT
< Etag: "588603eb-98b"
< Last-Modified: Mon, 23 Jan 2017 13:23:55 GMT
< Pragma: no-cache
< Server: bfe/1.0.8.18
< Set-Cookie: BDORZ=27315; max-age=86400; domain=.baidu.com; path=/
< 
<!DOCTYPE html>
<!--STATUS OK-->
<html>
<head>
...
    <title>百度一下,你就知道</title>
</head>
<body link=#0000cc>
    ...
</body>
</html>

2.3 URL

  • Web應用上的每一個資源都有惟一的地址進行定位,被稱爲URL(Uniform Resource Locatiors)統一資源定位符。
    1. URL中可包含額外的參數,會追加到URL的最後,以?開頭,以&分隔各個參數,如:http://www.web.com:8080/welcome/hello.html?name=jack&age=20
    2. Web服務器中的應用通常默認在80端口運行,端口號爲80的在URL中可省略不書寫。
      在這裏插入圖片描述

2.4 Servlet

  • Servlet是Server Applet的簡寫,也被稱爲運行在服務端的小程序,經過Servlet程序能夠生成動態Web內容,實現Java Web服務器功能的擴展。
  • Servlet須要依付於Servlet容器(好比Tomcat等)才能運行
    在這裏插入圖片描述

2.4.1 編寫第一個Servlet程序

  • Java Servlet是一個接口,定義了Java類被瀏覽器訪問到的規則。程序員

  • Java程序員自定義一個類,實現Servlet接口並重寫接口中的方法,Web服務器(如Tomcat等)就能夠識別並執行這個程序。web

  • 建立JavaEE項目
    在這裏插入圖片描述
    在這裏插入圖片描述小程序

  • 定義一個類,實現Servlet接口,實現或重寫接口中的方法瀏覽器

/**
 * HttpServlet
 *
 * @author zhuhuix
 * @date 2020-08-25
 */
public class HttpServletDemo extends HttpServlet {

    // 重寫Get請求方法
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        // 業務邏輯處理
        LocalDate date = LocalDate.now();

        // 動態輸出HTML頁面
        PrintWriter out = resp.getWriter();
        out.println("<html>" +
                "<body>" +
                "<h1>"+date.toString()+"</h1> " +
                "<h2>hello world</h2> " +
                "</body>" +
                "</html>"
        );
    }
}
  • 在web.xml配置Servlet
<?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="4.0">
    
     <!-- 配置Servlet -->
    <servlet>
        <servlet-name>http-demo</servlet-name>
        <servlet-class>demo.servlet.HttpServletDemo</servlet-class>
    </servlet>
    <servlet-mapping>
        <servlet-name>http-demo</servlet-name>
        <url-pattern>/http-demo</url-pattern>
    </servlet-mapping>
</web-app>
  • 運行
    在這裏插入圖片描述
  • 訪問http://localhost:8080/servlet_war_exploded/http-demo路徑
    在這裏插入圖片描述

2.5 JSP

  • JSP(Java Server Pages)也是一種Java servlet,主要用於實現Java web應用程序的用戶界面部分。經過結合HTML代碼、XHTML代碼、XML元素以及嵌入JSP操做和命令來編寫JSP。簡而言之,就是在HTML網頁中嵌入了Java。
  • JSP頁面能夠與處理業務邏輯的 Servlet 一塊兒使用,這種模式被Java servlet 模板引擎所支持
<%@ page import="java.time.LocalDate" %>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
  <body>
  <h1><%= LocalDate.now()%></h1>
  <h2>hello world!!!</h2>
  </body>
</html>
public class HttpServletDemo extends HttpServlet {
    // 重寫Get請求方法
   @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
    	// 調用jsp頁面
        req.getRequestDispatcher("/index.jsp").forward(req,resp);
    }
}

2.6 容器

  • 容器就是程序運行時須要的環境;好比Tomcat就是一個Servlet/JSP的容器。
  • Servlet 容器具有管理Servlet對象的功能。
    1. 通訊支持:利用容器提供的方法,能夠輕鬆實現Web服務器與Servlet進行通訊,無需本身創建ServerSocket、監聽端口,字節流傳輸等底層協議。
    2. 生命週期管理: 容器會負責Servlet的實例化、初始化、調用方法、釋放資源等控制。
    3. 多線程支持:容器接收到每一個Servlet請求時創建新的線程,並協調線程運行。
    4. 配置式管理:利用容器支持的xml進行配置式管理,在環境改變時,無需硬性修改程序及編譯程序。

在這裏插入圖片描述
在這裏插入圖片描述

2.7 URL映射到Servlet

  • 經過JavaEE的XML配置部署文件中的元素〈servlet〉〈servlet-mapping〉,把URL映射到對應的Servlet上。
    在這裏插入圖片描述

3、一個完整的Java Web基礎應用

在這裏插入圖片描述

在這裏插入圖片描述

/**
 * HttpServlet
 *
 * @author zhuhuix
 * @date 2020-08-25
 */
public class HttpServletDemo extends HttpServlet {

    // 重寫Get請求方法,返回一個表單頁面
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws IOException, ServletException {
        req.getRequestDispatcher("/form.jsp").forward(req, resp);
    }

    // 重寫Post請求方法,提交表單上的數據,並返回結果頁面
    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        String name = req.getParameter("name");
        Integer age = Integer.parseInt(req.getParameter("age"));
        if (name != null && age != null) {
            User user = new User(name, age);
            req.setAttribute("name", name);
            req.setAttribute("age", age);
            if (user.checkName()) {
                req.setAttribute("result","登記成功");
            }else {
                req.setAttribute("result","登記失敗:名稱中包含非法字符");
            }
            RequestDispatcher view = req.getRequestDispatcher("/result.jsp");
            view.forward(req, resp);
        }
    }
}
/**
 * 用戶類
 */
public class User {
    private String name;
    private Integer age;

    public User() {
    }

    public User(String name, Integer age) {
        this.name = name;
        this.age = age;
    }

    // 判斷名稱是否非法
    public Boolean checkName() {
        return !this.name.contains("admin");
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public Integer getAge() {
        return age;
    }

    public void setAge(Integer age) {
        this.age = age;
    }
}
<?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="4.0">

        <servlet>
            <servlet-name>http-demo</servlet-name>
            <servlet-class>demo.servlet.HttpServletDemo</servlet-class>
        </servlet>

        <servlet-mapping>
            <servlet-name>http-demo</servlet-name>
            <url-pattern>/http-demo</url-pattern>
        </servlet-mapping>


</web-app>
<!--表單頁面-->
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>登記用戶</title>
</head>
<body>
    <form method="post" action="http-demo">
        姓名:<input name="name" type="text"/><br>
        年齡:<input name="age" type="text"/><br>

            <input type="submit" title="提交" >

    </form>
</body>
</html>
<!--結果頁面-->
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>登記結果</title>
</head>
<body>
    <h2>姓名:${name} 年齡:${age} ${result} </h2><br>
</body>
</html>

在這裏插入圖片描述
在這裏插入圖片描述

4、小結

  • 以上三個小節,只對Java Web中的基礎要素及這些要素組成的最小體系結構作了回顧。
  • Java EE中Web開發的要點還有不少,好比Servlet的請求與響應的具體細節、JSTL語法,過濾器與監聽器,Session與Cookie等。
  • 在重度依賴各類框架開發的今天,謹以此文拋磚引玉,讓咱們先拋開框架,重拾基礎。
相關文章
相關標籤/搜索