Spring學習 6- Spring MVC (Spring MVC原理及配置詳解)

 百度的面試官問:Web容器,Servlet容器,SpringMVC容器的區別:

我還寫了個文章,說明web容器與servlet容器的聯繫,參考:servlet單實例多線程模式html

SpringMVC的Controller 是單實例的(阿里面試題);至於緣由前端

一、爲了性能。 單例不用每次都new,固然快了。java

二、不須要多例。只要controller中不定義屬性,那麼單例徹底是安全的,若是你給controller中定義不少的屬性,那麼單例確定會出現競爭訪問了。web

因此不要再controller中定義成員變量 ,若是定義了多個屬性,須要修改註解@Scope("prototype"),面試

參考:spring的controller默認是單例仍是多例spring

這個文章有web容器與springMVC的聯繫:數據庫

客戶端發出一個http請求給web服務器,web服務器對http請求進行解析,若是匹配DispatcherServlet的請求映射路徑(在web.xml中指定),web容器將請求轉交給DispatcherServletapache

具體看下面:設計模式

 

 

1.Spring MVC概述:

Spring MVC是Spring提供的一個強大而靈活的web框架。藉助於註解,Spring MVC提供了幾乎是POJO的開發模式,使得控制器的開發和測試更加簡單。這些控制器通常不直接處理請求,而是將其委託給Spring上下文中的其餘bean,經過Spring的依賴注入功能,這些bean被注入到控制器中。spring-mvc

Spring MVC主要由DispatcherServlet、處理器映射、處理器(控制器)、視圖解析器、視圖組成。他的兩個核心是兩個核心:

處理器映射:選擇使用哪一個控制器來處理請求 
視圖解析器:選擇結果應該如何渲染

經過以上兩點,Spring MVC保證瞭如何選擇控制處理請求和如何選擇視圖展示輸出之間的鬆耦合。

Dispatcher Servlet(Spring控制器)

在最簡單的Spring MVC應用程序中,控制器是惟一的你須要在Java web部署描述文件(即web.xml文件)中配置的Servlet。Spring MVC控制器 ——一般稱做Dispatcher Servlet,實現了前端控制器設計模式。而且每一個web請求必須經過它以便它可以管理整個請求的生命週期。

當一個web請求發送到Spring MVC應用程序,dispatcher servlet首先接收請求。而後它組織那些在Spring web應用程序上下文配置的(例如實際請求處理控制器和視圖解析器)或者使用註解配置的組件,全部的這些都須要處理該請求。 

在Spring3.0中定義一個控制器類,這個類必須標有@Controller註解。當有@Controller註解的控制器收到一個請求時,它會尋找一個合適的handler方法去處理這個請求。這就須要控制器經過一個或多個handler映射去把每一個請求映射到handler方法。爲了這樣作,一個控制器類的方法須要被@RequestMapping註解裝飾,使它們成爲handler方法。

handler方法處理完請求後,它把控制權委託給視圖名與handler方法返回值相同的視圖。爲了提供一個靈活的方法,一個handler方法的返回值並不表明一個視圖的實現而是一個邏輯視圖,即沒有任何文件擴展名。你能夠將這些邏輯視圖映射到正確的實現,並將這些實現寫入到上下文文件,這樣你就能夠輕鬆的更改視圖層代碼甚至不用修改請求handler類的代碼。
爲一個邏輯名稱匹配正確的文件是視圖解析器的責任。一旦控制器類已將一個視圖名稱解析到一個視圖實現。它會根據視圖實現的設計來渲染對應對象。 

2.SpringMVC運行原理

   1.springmvc請全部的請求都提交給DispatcherServlet,它會委託應用系統的其餘模塊負責負責對請求進行真正的處理工做。 

  2.DispatcherServlet查詢一個或多個HandlerMapping,找處處理請求的Controller. 

  3.DispatcherServlet請請求提交到目標Controller 

  4.Controller進行業務邏輯處理後,會返回一個ModelAndView 

  5.Dispathcher查詢一個或多個ViewResolver視圖解析器,找到ModelAndView對象指定的視圖對象 

  6.視圖對象負責渲染返回給客戶端。 

  

上面的是springMVC的工做原理圖:

一、客戶端發出一個http請求給web服務器,web服務器對http請求進行解析,若是匹配DispatcherServlet的請求映射路徑(在web.xml中指定),web容器將請求轉交給DispatcherServlet.

二、DipatcherServlet接收到這個請求以後將根據請求的信息(包括URL、Http方法、請求報文頭和請求參數Cookie等)以及HandlerMapping的配置找處處理請求的處理器(Handler)。

3-四、DispatcherServlet根據HandlerMapping找到對應的Controller,將處理權交給相應的Controller(Controller將具體的處理進行封裝),再由具體的HandlerAdapter對Handler進行具體的調用。

五、Controller對數據處理完成之後將返回一個ModelAndView()對象給DispatcherServlet。

六、Controller返回的ModelAndView()只是一個邏輯視圖並非一個正式的視圖,DispatcherSevlet經過ViewResolver將邏輯視圖轉化爲真正的視圖View。

七、Dispatcher經過model解析出ModelAndView()中的參數進行解析最終展示出完整的view並返回給客戶端。

 

3.SpringMVC接口解釋

(1)DispatcherServlet接口: 
Spring提供的前端控制器,全部的請求都有通過它來統一分發。在DispatcherServlet將請求分發給Spring Controller以前,須要藉助於Spring提供的HandlerMapping定位到具體的Controller。 
(2)HandlerMapping接口: 
可以完成客戶請求到Controller映射。 
(3)Controller接口: 
須要爲併發用戶處理上述請求,所以實現Controller接口時,必須保證線程安全而且可重用。 
Controller將處理用戶請求,這和Struts Action扮演的角色是一致的。一旦Controller處理完用戶請求,則返回ModelAndView對象給DispatcherServlet前端控制器,ModelAndView中包含了模型(Model)和視圖(View)。 
從宏觀角度考慮,DispatcherServlet是整個Web應用的控制器;從微觀考慮,Controller是單個Http請求處理過程當中的控制器,而ModelAndView是Http請求過程當中返回的模型(Model)和視圖(View)。 
(4)ViewResolver接口: 
Spring提供的視圖解析器(ViewResolver)在Web應用中查找View對象,從而將相應結果渲染給客戶。

4.DispatcherServlet:

是整個Spring MVC的核心。它負責接收HTTP請求組織協調Spring MVC的各個組成部分。其主要工做有如下三項: 
(1)截獲符合特定格式的URL請求。 
(2)初始化DispatcherServlet上下文對應WebApplicationContext,並將其與業務層、持久化層的WebApplicationContext創建關聯。 
(3)初始化Spring MVC的各個組成組件,並裝配到DispatcherServlet中。

5. SpringMVC配置

項目總體結構以下: 
 
(1)在web.xml文件中進行配置,在配置中設置springmvc-context.xml的路徑,代碼以下:

<servlet>
        <servlet-name>appServlet</servlet-name>
        <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
        <init-param>
            <param-name>contextConfigLocation</param-name>
            <param-value>classpath:META-INF/spring/springmvc-context.xml</param-value>
        </init-param>
        <load-on-startup>1</load-on-startup>
    </servlet>
    <servlet-mapping>
        <servlet-name>appServlet</servlet-name>
        <url-pattern>/</url-pattern>
    </servlet-mapping>

 

(2)配置springmvc-context.xml文件,這一部分主要是開啓註解功能、配置試圖解析器,代碼以下:

 
<?xml version="1.0" encoding="UTF-8"?>
<beans:beans xmlns:beans="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context"
    xmlns:mvc="http://www.springframework.org/schema/mvc" xmlns:p="http://www.springframework.org/schema/p"
    xsi:schemaLocation="http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-3.0.xsd
        http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
        http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.0.xsd ">
    <mvc:annotation-driven />
    <!-- ①:對web包中的全部類進行掃描,以完成Bean建立和自動依賴注入的功能 -->
    <context:component-scan base-package="com.zjn" />

    <!-- 這兩個類用來啓動基於Spring MVC的註解功能,將控制器與方法映射加入到容器中 -->
    <beans:bean
        class="org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping" />
    <beans:bean
        class="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter" />

    <!-- 這個類用於Spring MVC視圖解析 -->
    <beans:bean id="viewResolver"
        class="org.springframework.web.servlet.view.InternalResourceViewResolver">
        <beans:property name="prefix" value="/WEB-INF/pages/" />
        <beans:property name="suffix" value=".jsp" />
    </beans:bean>

</beans:beans>
 

 (3)配置文件完成了,下面開始寫代碼, 

兩個jsp界面: 
create.jsp

<%@ page language="java" contentType="text/html; charset=UTF-8"
    pageEncoding="UTF-8"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Add User From</title>
</head>
<body>
    <form action="save" method="post">
        <fieldset>
        <legend>建立用戶</legend>
            <p>
                <label>姓名:</label> <input type="text" id="name" name="name"
                    tabindex="1">
            </p>
            <p>
                <label>年齡:</label> <input type="text" id="age" name="age"
                    tabindex="2">
            </p>
            <p>
                <label>密碼:</label> <input type="text" id="pwd" name="pwd"
                    tabindex="3">
            </p>
            <p id="buttons">
                <input id="reset" type="reset" tabindex="4" value="取消"> <input
                    id="submit" type="submit" tabindex="5" value="建立">
            </p>
        </fieldset>
    </form>
</body>
</html>

 detail.jsp

<%@ page language="java" contentType="text/html; charset=UTF-8"
    pageEncoding="UTF-8"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Insert title here</title>
</head>
<body>
    <div id="gloobal">
        <h4>建立成功</h4>
        <p>
        <h5>詳情:</h5>
        姓名:${user.name}<br /> 年齡:${user.age}<br /> 密碼:${user.pwd}<br />
        </p>
    </div>
</body>
</html>

 UserController.Java

package com.zjn.controller;

import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.ModelAttribute;
import org.springframework.web.bind.annotation.RequestMapping;

import com.zjn.entity.User;

/**
 * 用戶管理
 * 
 * @author zjn
 */
@Controller
public class UserController {

    @RequestMapping("")
    public String Create(Model model) {
        return "create";
    }

    @RequestMapping("/save")
    public String Save(@ModelAttribute("form") User user, Model model) { // user:視圖層傳給控制層的表單對象;model:控制層返回給視圖層的對象
        model.addAttribute("user", user);
        return "detail";
    }
}
User.java
package com.zjn.entity;

import java.io.Serializable;
import java.util.Date;

public class User implements Serializable {
    /**
     * @author zjn
     */
    private static final long serialVersionUID = 1L;
    private Integer id; // id
    private String name; // name
    private String pwd; // pwd
    private Integer age; // age
    private Date creatTime; // creatTime

    public Integer getId() {
        return id;
    }

    public void setId(Integer id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

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

    public String getPwd() {
        return pwd;
    }

    public void setPwd(String pwd) {
        this.pwd = pwd;
    }

    public Integer getAge() {
        return age;
    }

    public void setAge(Integer age) {
        this.age = age;
    }

    public Date getCreatTime() {
        return creatTime;
    }

    public void setCreatTime(Date creatTime) {
        this.creatTime = creatTime;
    }

}

 (4)運行結果

初始頁面: 

輸入參數: 

點擊建立: 
 

MVC框架是什麼

模型-視圖-控制器(MVC)是一個衆所周知的以設計界面應用程序爲基礎的設計模式。它主要經過分離模型、視圖及控制器在應用程序中的角色將業務邏輯從界面中解耦。一般,模型負責封裝應用程序數據在視圖層展現。視圖僅僅只是展現這些數據,不包含任何業務邏輯。控制器負責接收來自用戶的請求,並調用後臺服務(manager或者dao)來處理業務邏輯。處理後,後臺業務層可能會返回了一些數據在視圖層展現。控制器收集這些數據及準備模型在視圖層展現。MVC模式的核心思想是將業務邏輯從界面中分離出來,容許它們單獨改變而不會相互影響。

在Spring MVC應用程序中,模型一般由POJO對象組成,它在業務層中被處理,在持久層中被持久化。視圖一般是用JSP標準標籤庫(JSTL)編寫的JSP模板。控制器部分是由dispatcher servlet負責,在本教程中咱們將會了解更多它的相關細節。
一些開發人員認爲業務層和DAO層類是MVC模型組件的一部分。我對此持有不一樣的意見。我不認爲業務層及DAO層類爲MVC框架的一部分。一般一個web應用是3層架構,即數據-業務-表示。MVC其實是表示層的一部分。

在這個應用程序中,我將建立最簡單的員工管理應用程序的演示,它只有一個功能,即系統提供的全部僱員的列表。讓咱們記下此應用程序的目錄結構。

如今讓咱們編寫全部涉及這個入門應用程序的文件。

pom.xml

如下pom.xml文件中包含spring mvc依賴及爲編寫jsp文件提供支持的taglib相關依賴 

 

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <groupId>com.howtodoinjava.demo</groupId>
    <artifactId>springmvcexample</artifactId>
    <packaging>war</packaging>
    <version>1.0-SNAPSHOT</version>
    <name>springmvcexample Maven Webapp</name>
    <url>http://maven.apache.org</url>
    <dependencies>
 
        <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
            <version>4.12</version>
            <scope>test</scope>
        </dependency>
 
        <!-- Spring MVC support -->
 
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-webmvc</artifactId>
            <version>4.1.4.RELEASE</version>
        </dependency>
 
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-web</artifactId>
            <version>4.1.4.RELEASE</version>
        </dependency>
 
        <!-- Tag libs support for view layer -->
 
        <dependency>
            <groupId>javax.servlet</groupId>
            <artifactId>jstl</artifactId>
            <version>1.2</version>
            <scope>runtime</scope>
        </dependency>
 
        <dependency>
            <groupId>taglibs</groupId>
            <artifactId>standard</artifactId>
            <version>1.1.2</version>
            <scope>runtime</scope>
        </dependency>
 
    </dependencies>
 
    <build>
        <finalName>springmvcexample</finalName>
    </build>
</project>

 

這最精簡的web.xml文件聲明瞭一個Servlet(即dispatcher servlet)來接收全部類型的請求。Dispatcher servlet在這裏充當前端控制器的角色。 

 

<web-app id="WebApp_ID" version="2.4"
    xmlns="http://java.sun.com/xml/ns/j2ee"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee
 
http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd">
 
    <display-name>Spring Web MVC Hello World Application</display-name>
 
    <servlet>
        <servlet-name>spring</servlet-name>
            <servlet-class>
                org.springframework.web.servlet.DispatcherServlet
            </servlet-class>
        <load-on-startup>1</load-on-startup>
    </servlet>
 
    <servlet-mapping>
        <servlet-name>spring</servlet-name>
        <url-pattern>/</url-pattern>
    </servlet-mapping>
 
</web-app>

 

spring-servlet.xml(你也能夠用applicationContext.xml文件)

在請求handler、業務層、dao層,咱們使用帶註解的類,因此我爲「com.howtodoinjava.demo」包裏的全部類啓用了註解處理 

 

<web-app id="WebApp_ID" version="2.4"
    xmlns="http://java.sun.com/xml/ns/j2ee"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee
 
http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd">
 
    <display-name>Spring Web MVC Hello World Application</display-name>
 
    <servlet>
        <servlet-name>spring</servlet-name>
            <servlet-class>
                org.springframework.web.servlet.DispatcherServlet
            </servlet-class>
        <load-on-startup>1</load-on-startup>
    </servlet>
 
    <servlet-mapping>
        <servlet-name>spring</servlet-name>
        <url-pattern>/</url-pattern>
    </servlet-mapping>
 
</web-app>

 

註解@RequestMapping在類級別和方法級別層面肯定將被調用方法的URL。  

<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:context="http://www.springframework.org/schema/context"
    xsi:schemaLocation="http://www.springframework.org/schema/beans
 
http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
 
 
http://www.springframework.org/schema/context
 
 
http://www.springframework.org/schema/context/spring-context-3.0.xsd">
 
    <context:component-scan base-package="com.howtodoinjava.demo" />
 
    <bean class="org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping" />
    <bean class="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter" />
 
    <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
        <property name="prefix" value="/WEB-INF/views/" />
        <property name="suffix" value=".jsp" />
    </bean>
 
</beans>

閱讀更多:如何使用@Component、@Respository、@Service及@Controller註解?

EmployeeVO.java

這個類做爲MVC模式的模型。 

package com.howtodoinjava.demo.model;
 
import java.io.Serializable;
 
public class EmployeeVO implements Serializable
{
    private static final long serialVersionUID = 1L;
 
    private Integer id;
    private String firstName;
    private String lastName;
 
    //Setters and Getters
 
    @Override
    public String toString() {
        return "EmployeeVO [id=" + id + ", firstName=" + firstName
                + ", lastName=" + lastName + "]";
    }
}

 

EmployeeDAO.java

這個類位於三層架構中的第三層。負責與底層的數據庫存儲進行交互。  

import java.util.List;
 
import com.howtodoinjava.demo.model.EmployeeVO;
 
public interface EmployeeDAO
{
    public List<EmployeeVO> getAllEmployees();
}

EmployeeDAOImpl.java 

import java.util.ArrayList;
import java.util.List;
 
import org.springframework.stereotype.Repository;
 
import com.howtodoinjava.demo.model.EmployeeVO;
 
@Repository
public class EmployeeDAOImpl implements EmployeeDAO {
 
    public List<EmployeeVO> getAllEmployees()
    {
        List<EmployeeVO> employees = new ArrayList<EmployeeVO>();
 
        EmployeeVO vo1 = new EmployeeVO();
        vo1.setId(1);
        vo1.setFirstName("Lokesh");
        vo1.setLastName("Gupta");
        employees.add(vo1);
 
        EmployeeVO vo2 = new EmployeeVO();
        vo2.setId(2);
        vo2.setFirstName("Raj");
        vo2.setLastName("Kishore");
        employees.add(vo2);
 
        return employees;
    }
}

EmployeeManager.java

這個類處於三層架構中的第二層。負責與DAO層交互。 

import java.util.List;
 
import com.howtodoinjava.demo.model.EmployeeVO;
 
public interface EmployeeManager
{
    public List<EmployeeVO> getAllEmployees();
}

 

EmployeeManagerImpl.java 

import java.util.List;
 
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
 
import com.howtodoinjava.demo.dao.EmployeeDAO;
import com.howtodoinjava.demo.model.EmployeeVO;
 
@Service
public class EmployeeManagerImpl implements EmployeeManager {
 
    @Autowired
    EmployeeDAO dao;
 
    public List<EmployeeVO> getAllEmployees()
    {
        return dao.getAllEmployees();
    }
}

employeesListDisplay.jsp

這個jsp被用於顯示系統中的全部員工。它循環遍歷employee集合,而且在一個表中打印他們的詳細信息。這符合MVC模式的視圖層。 

<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core"%>
<%@ taglib prefix="fmt" uri="http://java.sun.com/jsp/jstl/fmt"%>
 
<html>
<head>
    <title>Spring MVC Hello World</title>
</head>
 
<body>
    <h2>All Employees in System</h2>
 
    <table border="1">
        <tr>
            <th>Employee Id</th>
            <th>First Name</th>
            <th>Last Name</th>
        </tr>
        <c:forEach items="${employees}" var="employee">
            <tr>
                <td>${employee.id}</td>
                <td>${employee.firstName}</td>
                <td>${employee.lastName}</td>
            </tr>
        </c:forEach>
    </table>
 
</body>
</html>

 

如今在您的應用程序服務器(我用的是tomcat 7)部署應用程序。並點擊「http://localhost:8080/springmvcexample/employee-module/getAllEmployees」。若是你已正確配置全部內容,你將會在屏幕下看到:

參考:源碼下載 

參考:Spring MVC 入門示例講解

下載地址: SpringMVC項目源碼下載

參考:Spring MVC原理及配置詳解

相關文章
相關標籤/搜索