Spring入門(十二):Spring MVC使用講解

  • 2019 年 10 月 3 日
  • 筆記

1. Spring MVC介紹

提到MVC,參與過Web應用程式開發的同學都很熟悉,它是展現層(也可以理解成直接展現給用戶的那一層)開發的一種架構模式,M全稱是Model,指的是數據模型,V全稱是View,指的是視圖頁面,如JSP、Thymeleaf等,C全稱是Controller,指的是控制器,用來處理用戶在客戶端(瀏覽器)發起的請求。

Spring MVC就是基於MVC模式實現,能夠幫你構建像Spring框架那樣靈活和松耦合的Web應用程式。

也就是說,Spring MVC是基於Spring的,它主要用來開發Web應用程式。

2. Spring MVC請求流程

在Web應用程式中,請求是必不可少的,每次請求由用戶(客戶端)發起,到用戶(客戶端)接收到響應結束,下圖展示了一個請求在Spring MVC中所經歷的每個流程:

接下來我們對圖中的每個流程做下講解:

  1. 請求離開瀏覽器時(圖中的①),會帶有用戶所請求內容的資訊,比如請求的URL和提交的表單資訊。
  2. 然後進入到請求的第一站,即Spring的DispatcherServlet,它的任務是將請求發送給Spring MVC控制器。
  3. 因為在應用程式中會有多個控制器,因此DispatcherServlet會查詢一個或多個處理器映射(handler mapping),處理器映射會根據請求所攜帶的URL來確定應該將請求發送給哪個控制器(圖中的②)。
  4. 確定好控制器後,DispatcherServlet會將請求發送給確定好的控制器,然後等待控制器處理用戶提交的資訊,不過一般情況下,控制器本身幾乎不處理工作,而是將業務邏輯委託給一個或多個服務對象進行處理(圖中的③)。
  5. 控制器完成業務邏輯處理後,通常會產生一些模型(Model)資訊,這些資訊需要返回給用戶並在瀏覽器上顯示,為了更友好的展示這些資訊,比如以Html形式展示,我們需要將資訊發送給一個視圖(View),比如JSP、Thymeleaf。
  6. 控制器所做的最後一件事就是將模型數據打包,並且標示出用於渲染輸出的視圖名,它會將請求連同模型和視圖名發送回DispatcherServlet(圖中的④),不過控制器只會返回一個視圖的邏輯名稱,而不是返回具體的某個特定視圖,這個邏輯名稱將會用來查找產生結果的真正視圖。DispatcherServlet會使用視圖解析器(view resolver)來將邏輯視圖名匹配為某個特定視圖(圖中的⑤),比如JSP或者Thymeleaf。
  7. 請求的最後一站是視圖的實現(圖中的⑥),在這裡視圖將使用模型數據渲染輸出,這個輸出會通過響應對象傳遞給用戶/客戶端(圖中的⑦)。

了解了Spring MVC的請求流程後,我們來接著了解下如何搭建和配置Spring MVC項目。

3. 搭建Spring MVC項目

我們仍然延用之前部落格中新建的spring-action項目,有興趣的同學可以看下本系列的前11篇部落格或者直接下載源碼:https://github.com/zwwhnly/spring-action.git

3.1 添加依賴

要想使用Spring MVC,首先我們需要在pom.xml中添加如下依賴:

<dependency>      <groupId>org.springframework</groupId>      <artifactId>spring-webmvc</artifactId>      <version>4.3.18.RELEASE</version>  </dependency>    <!-- 其他Web依賴 -->  <dependency>      <groupId>javax.servlet</groupId>      <artifactId>jstl</artifactId>      <version>1.2</version>  </dependency>  <dependency>      <groupId>javax.servlet</groupId>      <artifactId>javax.servlet-api</artifactId>      <version>4.0.1</version>      <scope>provided</scope>  </dependency>

因為後面要將項目部署到Tomcat,所以我們在pom.xml中配置下打包方式為war包:

<packaging>war</packaging>    <build>      <plugins>          <!--其他配置-->          <plugin>              <groupId>org.apache.maven.plugins</groupId>              <artifactId>maven-war-plugin</artifactId>              <version>3.2.3</version>              <configuration>                  <failOnMissingWebXml>false</failOnMissingWebXml>              </configuration>          </plugin>      </plugins>  </build>

3.2 新建演示頁面

在src/main/resources下新建views目錄,然後在此目錄下新建index.jsp頁面如下所示:

<%@ page contentType="text/html;charset=UTF-8" language="java" pageEncoding="UTF-8" %>  <html>  <head>      <title>Spring MVC</title>  </head>  <body>      <pre>          Welcome to Spring MVC world      </pre>  </body>  </html>

這裡可能存在的問題是,右鍵新建JSP文件時,沒有JSP文件模板,就像下面這樣:

解決方案如下所示:

依次點擊File–Project Structure,打開Project Structure對話框,左側選中Modules,然後點擊+號,選擇Web

此時再次右鍵新增JSP文件,就會看到JSP文件模板:

此時項目根目錄會生成1個web文件夾,可以將其刪除。

3.3 Spring MVC配置

新建配置類MyMvcConfig如下所示:

package chapter05.config;    import org.springframework.context.annotation.Bean;  import org.springframework.context.annotation.ComponentScan;  import org.springframework.context.annotation.Configuration;  import org.springframework.web.servlet.config.annotation.EnableWebMvc;  import org.springframework.web.servlet.view.InternalResourceViewResolver;  import org.springframework.web.servlet.view.JstlView;    /**   * Spring MVC配置   */  @Configuration  @EnableWebMvc  @ComponentScan("chapter05")  public class MyMvcConfig {      /**       * 視圖解析器配置       *       * @return       */      @Bean      public InternalResourceViewResolver viewResolver() {          InternalResourceViewResolver viewResolver = new InternalResourceViewResolver();            viewResolver.setPrefix("/WEB-INF/classes/views/");          viewResolver.setSuffix(".jsp");          viewResolver.setViewClass(JstlView.class);            return viewResolver;      }  }

注意事項:

1)該配置類使用了@EnableWebMvc註解來啟用Spring MVC,它會開啟一些默認配置。

2)該配置類配置了視圖解析器,這裡我們配置的是JSP的視圖解析器。

視圖解析器中,我們設置了前綴和後綴,如果控制器中返回的邏輯視圖名稱是index,實際渲染時找的視圖就是/WEB-INF/classes/views/index.jsp,為什麼設置的前綴是/WEB-INF/classes/views/而不是/src/main/resources/views/呢,那是因為項目編譯完運行時的目錄是/WEB-INF/classes/views/。

如果編譯完成該目錄下沒有jsp文件,則需要在pom.xml中添加如下配置:

<build>      <resources>          <resource>              <directory>src/main/resources</directory>              <includes>                  <include>**/*.jsp</include>                  <include>**/*.js</include>              </includes>          </resource>      </resources>  </build>

3.4 Web配置

新建Web配置類WebInitializer如下所示:

package chapter05.config;    import org.springframework.web.WebApplicationInitializer;  import org.springframework.web.context.support.AnnotationConfigWebApplicationContext;  import org.springframework.web.servlet.DispatcherServlet;    import javax.servlet.ServletContext;  import javax.servlet.ServletException;  import javax.servlet.ServletRegistration.Dynamic;    public class WebInitializer implements WebApplicationInitializer {      @Override      public void onStartup(ServletContext servletContext) throws ServletException {          AnnotationConfigWebApplicationContext context = new AnnotationConfigWebApplicationContext();          context.register(MyMvcConfig.class);          context.setServletContext(servletContext);            Dynamic servlet = servletContext.addServlet("dispatcher", new DispatcherServlet(context));          servlet.addMapping("/");          servlet.setLoadOnStartup(1);      }  }

這裡重點要關注的是,該類實現了介面WebApplicationInitializer並重寫了onStartup()方法,WebApplicationInitializer類是Spring提供用來配置Servlet 3.0+版本配置的介面,從而可以替代掉web.xml。

3.5 新建控制器

新建控制器HelloController如下所示:

package chapter05.controller;    import org.springframework.stereotype.Controller;  import org.springframework.web.bind.annotation.RequestMapping;    @Controller  public class HelloController {      @RequestMapping("/index")      public String hello() {          // 這裡返回的邏輯視圖名          return "index";      }  }

上面程式碼中的@Controller註解聲明這是1個控制器,@RequestMapping("/index")用來配置URL映射,現在問題來了,我們如何查看程式碼的運行效果呢?

這就涉及到了項目打包和項目部署,我們繼續往下看。

4. 項目部署到Tomcat中

4.1 打包(war包)

因為我們的項目是通過Maven管理的,在pom.xml中也配置了打包方式為war包和打包插件,所以我們可以在IDEA的右側找到Maven資源管理器,然後如下圖所示,點擊clean:

然後再點擊package:

最後生成的war包如下所示:

這就是我們要部署到Tomcat中的war包。

4.2 Tomcat安裝及部署

既然要部署到Tomcat中,那麼就有2個問題需要解答:

  1. 什麼是Tomcat?
  2. Tomcat如何安裝?

先回答第1個問題,Tomcat是一個免費的開放源程式碼的輕量級的Web應用伺服器,如果你接觸過.NET的Web開發,它就類似於IIS。

再回答第2個問題,可以參考如下步驟安裝Tomcat。

打開Tomcat官網,找到你要下載的Tomcat版本,我這裡選擇的是Tomcat 8.5.45 Released版本:

然後選擇合適的版本下載,因為我的開發機器是Windows 64位作業系統,所以我選擇的是如下所示的版本:

下載完成後,將其解壓到你喜歡的目錄,我解壓到的目錄是E:Toolsapache-tomcat-8.5.45-windows-x64apache-tomcat-8.5.45,解壓完成後長如下這樣:

其中webapps就是網站要部署的目錄。

安裝完成後,考慮的問題就是如何啟動Tomcat?

第1種方法是雙擊bin目錄下的tomcat8.exe:

然後在瀏覽器輸入地址http://localhost:8080/,看到如下介面,代表Tomcat安裝部署成功。

使用這種方法的缺點就是,如果把tomcat8.exe打開的窗口關閉了,Tomcat也就關閉了,非常不方便,因此建議使用第2種方法,將Tomcat安裝成一個後台服務,讓其在後台運行,操作方法如下所示:

依次打開電腦–屬性–高級系統設置–高級–環境變數,新增系統變數:

變數名:CATALINA_HOME

變數值:E:Toolsapache-tomcat-8.5.45-windows-x64apache-tomcat-8.5.45(你將Tomcat解壓的目錄)

然後編輯系統變數Path,在其最後添加如下內容:

;%CATALINA_HOME%lib;%CATALINA_HOME%bin

然後以管理員身份打開cmd窗口,切換到Tomcat的bin目錄,執行命令:service.bat install。

然後打開Windows的服務列表,會看到一個Tomcat8的服務:

將服務修改成自動啟動並啟動該服務即可在後台一直運行Tomcat伺服器。

如果你好奇新建的系統變數為什麼必須是CATALINA_HOME,那麼可以用記事本打開bin目錄下的service.bat,看到如下內容你就明白了:

從上圖也可以看出,運行Tomcat需要依賴環境變數JAVA_HOME(配置JAVA SDK路徑),不過我之前已經配置過了,如下所示:

4.3 將war包部署到Tomcat中

將之前打包好的spring-action-1.0-SNAPSHOT.war複製到Tomcat的webapps目錄:

因為我們的Tomcat伺服器設置成了後台運行,過一會該目錄就會生成1個和war包名相同的spring-action-1.0-SNAPSHOT文件夾,目錄結構如下所示:

我們的程式碼和JSP視圖文件都在WEB-INF目錄下的classes文件夾下:

在瀏覽器中輸入地址http://localhost:8080/spring-action-1.0-SNAPSHOT/index,頁面展示如下所示:

5. 源碼及參考

源碼地址:https://github.com/zwwhnly/spring-action.git,歡迎下載。

Craig Walls 《Spring實戰(第4版)》

汪雲飛《Java EE開發的顛覆者:Spring Boot實戰》

【IntelliJ IDEA】使用idea解決新建jsp文件而找不到jsp文件模版的新建選項

Tomcat安裝及配置教程

Tomcat安裝及後台運行的方法

6. 最後

歡迎掃碼關注微信公眾號:「申城異鄉人」,定期分享Java技術乾貨,讓我們一起進步。