20201127黄春跃

20201127黄春跃

知识点

SpringMVC 是什么

SpringMVC 是一种基于 Java 的实现 MVC 设计模型的请求驱动类型的轻量级 Web 框架,属于
SpringFrameWork 的后续产品,已经融合在 Spring Web Flow 里面。Spring 框架提供了构建 Web 应
用程序的全功能 MVC 模块。使用 Spring 可插入的 MVC 架构,从而在使用 Spring 进行 WEB 开发时,
可以选择使用 Spring的 Spring MVC 框架或集成其他 MVC 开发框架,如 Struts1(现在一般不用),
Struts2 等。
SpringMVC 已经成为目前最主流的 MVC 框架之一,并且随着 Spring3.0 的发布,全面超越 Struts2,
成为最优秀的 MVC 框架。
它通过一套注解,让一个简单的 Java 类成为处理请求的控制器,而无须实现任何接口。同时它还支持
RESTful 编程风格的请求。

SpringMVC 在三层架构的位置

SpringMVC 的优势

清晰的角色划分:
前端控制器( DispatcherServlet )
请求到处理器映射( HandlerMapping )
处理器适配器( HandlerAdapter )
视图解析器( ViewResolver )
处理器或页面控制器( Controller )
验证器( Validator )
命令对象( Command 请求参数绑定到的对象就叫命令对象)表单对象( Form Object 提供给表单展示和提交到的对象就叫表单对象)。

分工明确,而且扩展点相当灵活,可以很容易扩展,虽然几乎不需要。

由于命令对象就是一个 POJO,无需继承框架特定 API,可以使用命令对象直接作为业务对象。

和 Spring 其他框架无缝集成,是其它 Web 框架所不具备的。

可适配,通过 HandlerAdapter 可以支持任意的类作为处理器。

可定制性,HandlerMapping、ViewResolver 等能够非常简单的定制。

功能强大的数据验证、格式化、绑定机制。

利用 Spring 提供的 Mock 对象能够非常简单的进行 Web 层单元测试。

本地化、主题的解析的支持,使我们更容易进行国际化和主题的切换。

强大的 JSP 标签库,使 JSP 编写更容易。
………………还有比如RESTful风格的支持、简单的文件上传、约定大于配置的契约式编程支持、基于注解
的零配置支持等等。

SpringMVC 和 Struts2 的优略分析

共同点:
它们都是表现层框架,都是基于 MVC 模型编写的。
它们的底层都离不开原始 Servlet API。
它们处理请求的机制都是一个核心控制器。
区别:
Spring MVC 的入口是 Servlet, 而 Struts2 是 Filter
Spring MVC 是基于方法设计的,而 Struts2 是基于类,Struts2 每次执行都会创建一个动作类。
所以 Spring MVC 会稍微比 Struts2 快些。
Spring MVC 使用更加简洁,同时还支持 JSR303, 处理 ajax 的请求更方便
(JSR303 是一套 JavaBean 参数校验的标准,它定义了很多常用的校验注解,我们可以直接将这些
注解加在我们 JavaBean 的属性上面,就可以在需要校验的时候进行校验了。)
Struts2 的 OGNL 表达式使页面的开发效率相比 Spring MVC 更高些,但执行效率并没有比 JSTL
提升,尤其是 struts2 的表单标签,远没有 html 执行效率高。

SpringMVC 的入门

前期准备

  1. 创建一个 maven web 工程

  2. 创建一个 jsp 用于发送请求

    index.jsp

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
      <title>SpringMVC 入门案例</title>
</head>
<body>
<a href="${pageContext.request.contextPath}/hello">SpringMVC 入门案例</a>
<br/>
<a href="hello">SpringMVC 入门案例</a>
</body>
</html>

导入依赖

<properties>
  <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
  <maven.compiler.source>1.8</maven.compiler.source>
  <maven.compiler.target>1.8</maven.compiler.target>
  <spring.version>5.0.2.RELEASE</spring.version>
 </properties>
 <dependencies>
  <dependency>
   <groupId>org.springframework</groupId>
   <artifactId>spring-context</artifactId>
   <version>${spring.version}</version>
  </dependency>
  <!-- springMvc jar-->
  <dependency>
   <groupId>org.springframework</groupId>
   <artifactId>spring-web</artifactId>
   <version>${spring.version}</version>
  </dependency>
  <dependency>
   <groupId>org.springframework</groupId>
   <artifactId>spring-webmvc</artifactId>
   <version>${spring.version}</version>
  </dependency>
  <!-- 编译器servlet支持 -->
  <dependency>
   <groupId>javax.servlet</groupId>
   <artifactId>servlet-api</artifactId>
   <version>2.5</version>
   <scope>provided</scope>
  </dependency>
  <dependency>
   <groupId>javax.servlet.jsp</groupId>
   <artifactId>jsp-api</artifactId>
   <version>2.0</version>
   <scope>provided</scope>
  </dependency>
 </dependencies>

配置核心控制器-一个 Servlet

在 web.xml 配置文件中核心控制器 DispatcherServlet

<?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">
    <display-name>Archetype Created Web Application</display-name>
    <servlet>
        <servlet-name>SpringMVCDispatcherServlet</servlet-name>
        <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
        <!-- 配置初始化参数,用于读取 SpringMVC 的配置文件 -->
        <init-param>
            <param-name>contextConfigLocation</param-name>
            <param-value>classpath:springMVC.xml</param-value>
        </init-param>
        <!-- 配置 servlet 的对象的创建时间点:应用加载时创建。
    取值只能是非 0 正整数,表示启动顺序 -->
        <load-on-startup>1</load-on-startup>
    </servlet>
    <servlet-mapping>
        <servlet-name>SpringMVCDispatcherServlet</servlet-name>
        <url-pattern>/</url-pattern>
    </servlet-mapping>
</web-app>

创建 spring mvc 的配置文件

编写 springMVC.xml 的配置文件

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
           xmlns:mvc="http://www.springframework.org/schema/mvc"
           xmlns:context="http://www.springframework.org/schema/context"
           xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
           xsi:schemaLocation="http://www.springframework.org/schema/beans
           http://www.springframework.org/schema/beans/spring-beans.xsd
           http://www.springframework.org/schema/mvc
           http://www.springframework.org/schema/mvc/spring-mvc.xsd
           http://www.springframework.org/schema/context
           http://www.springframework.org/schema/context/spring-context.xsd">
          <!-- 配置创建 spring 容器要扫描的包 -->
          <context:component-scan base-package="com.itlaobing"/>
          <!-- 配置视图解析器 -->
          <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
            <!-- 视图前缀 -->
            <property name="prefix" value="/WEB-INF/pages/"></property>
            <!-- 视图后缀 -->
            <property name="suffix" value=".jsp"></property>
          </bean>
        </beans>

编写控制器并使用注解配置

package com.itlaobing.contorller;

import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;

@Controller

public class HelloController {
    @RequestMapping("/hello")
  public String sayHello() {
    System.out.println("HelloController 的 sayHello 方法执行了。。。。");
    return "success";
 }
}

在WEB-INF目录下创建pages文件夹,编写 success.jsp 的成功页面

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
  <title>success</title>
</head>
<body>
<h3>入门成功!!</h3>
</body>
</html>

测试

RequestMapping 注解

使用示例

package com.itlaobing.contorller;

import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
@Controller
@RequestMapping("/account")
public class AccountController {
    @RequestMapping("/findAccount")
    public String findAccount() {
        System.out.println("查询了账户。。。。");
        return "success2";
 }
    @RequestMapping("/saveAccount")
    public String saveAccount() {
        System.out.println("保存了账户。。。。");
        return "success3";
    }
    @RequestMapping(value = "/removeAccount" ,params = {"accountName","money"})
    public String removeAccount() {
        System.out.println("删除了账户。。。。");
        return "success2";
 }
}

method 属性的示例:

<%--
  Created by IntelliJ IDEA.
  User: 联想
  Date: 2020/11/27
  Time: 14:24
  To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>success</title>
</head>
<body>
<!-- 第一种访问方式 -->
    <a href="${pageContext.request.contextPath}/account/findAccount">查询账户</a>
    <br/>
    <!-- 第二种访问方式 -->
    <a href="account/findAccount">查询账户</a>
</body>
</html>

<%--
  Created by IntelliJ IDEA.
  User: 联想
  Date: 2020/11/27
  Time: 19:39
  To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
</head>
<body>
<!-- 请求方式的示例 -->
    <a href="account/saveAccount">保存账户,get 请求</a>
    <br/>
    <form action="${pageContext.request.contextPath}/account/saveAccount" method="post">
      <input type="submit" value="保存账户,post 请求">
    </form>
</body>
</html>

params 属性的示例:

@RequestMapping(value="/removeAccount",params= {"accountName","money"})
  public String removeAccount() {
    System.out.println("删除了账户");
    return "success2";
 }

请求参数的绑定

绑定的机制

我们都知道,表单中请求参数都是基于 key=value 的。
SpringMVC 绑定请求参数的过程是通过把表单提交请求参数,作为控制器中方法参数进行绑定的。

重新创建一个AccountControllerTest类,里面写一个public String findAccount(Integer accountId)方法

package com.itlaobing.contorller;

import org.springframework.web.bind.annotation.RequestMapping;

@RequestMapping("/findAccount2")
public class AccountControllerTest {
    public String findAccount(Integer accountId) {
        System.out.println("查询了账户。。。。" + accountId);
        return "findAccount";
    }
}

创建一个findAccount.jsp

<%--
  Created by IntelliJ IDEA.
  User: 联想
  Date: 2020/11/27
  Time: 20:30
  To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
</head>
<body>
    <a href="account/findAccount2?accountId=10">查询账户</a>
</body>
</html>

点击后

支持的数据类型:

基本类型参数:
包括基本类型和 String 类型
POJO 类型参数:
包括实体类,以及关联的实体类
数组和集合类型参数:
包括 List 结构和 Map 结构的集合(包括数组)
SpringMVC 绑定请求参数是自动实现的,但是要想使用,必须遵循使用要求。

使用要求:

如果是基本类型或者 String 类型:
要求我们的参数名称必须和控制器中方法的形参名称保持一致。(严格区分大小写)
如果是 POJO 类型,或者它的关联对象:
要求表单中参数名称和 POJO 类的属性名称保持一致,并且要有getter/setter方法。控制器方法的
参数类型是 POJO 类型。
如果是集合类型,有两种方式:
第一种:

要求集合类型的请求参数必须在 POJO 中。在表单中请求参数名称要和 POJO 中集合属性名称相
同。
给 List 集合中的元素赋值,使用下标。
给 Map 集合中的元素赋值,使用键值对。
第二种:
接收的请求参数是 json 格式数据。需要借助一个注解实现。
注意:
它还可以实现一些数据类型自动转换。内置转换器全都在:
org.springframework.core.convert.support 包下.有:

java.lang.Boolean -> java.lang.String : ObjectToStringConverter
java.lang.Character -> java.lang.Number : CharacterToNumberFactory
java.lang.Character -> java.lang.String : ObjectToStringConverter
java.lang.Enum -> java.lang.String : EnumToStringConverter
java.lang.Number -> java.lang.Character : NumberToCharacterConverter
java.lang.Number -> java.lang.Number : NumberToNumberConverterFactory
java.lang.Number -> java.lang.String : ObjectToStringConverter
java.lang.String -> java.lang.Boolean : StringToBooleanConverter
java.lang.String -> java.lang.Character : StringToCharacterConverter
java.lang.String -> java.lang.Enum : StringToEnumConverterFactory
java.lang.String -> java.lang.Number : StringToNumberConverterFactory
java.lang.String -> java.util.Locale : StringToLocaleConverter
java.lang.String -> java.util.Properties : StringToPropertiesConverter
java.lang.String -> java.util.UUID : StringToUUIDConverter
java.util.Locale -> java.lang.String : ObjectToStringConverter
java.util.Properties -> java.lang.String : PropertiesToStringConverter
java.util.UUID -> java.lang.String : ObjectToStringConverter
......
如遇特殊类型转换要求,需要我们自己编写自定义类型转换器。

使用示例

基本类型 和 String 类型作为参数

package com.itlaobing.contorller;

import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
@Controller
public class AccountControllerTest {
    //基本类型 和 String 类型作为参数
    @RequestMapping("/findAccount2")
  public String findAccount(Integer accountId, String accountName) {
    System.out.println("查询了账户。。。。" + accountId + "," + accountName);
    return "findAccounts";
 }

}

<%--
  Created by IntelliJ IDEA.
  User: 联想
  Date: 2020/11/27
  Time: 20:30
  To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
</head>
<body>
    <a href="findAccount2?accountId=10&accountName=zhangshan">查询账户</a>
</body>
</html>

POJO 类型作为参数

Account

package com.itlaobing.dao;

import lombok.Data;

import java.io.Serializable;

@Data
public class Account implements Serializable {
    private Integer id;
    private String name;
    private Float money;
    private Address address;
}

Address

package com.itlaobing.dao;

import lombok.Data;

import java.io.Serializable;

@Data
public class Address implements Serializable {
    private String provinceName;
    private String cityName;
}

 @RequestMapping("/saveAccount")
    public String saveAccount(Account account) {
        System.out.println("保存了账户。。。。" + account.getName()+account.getMoney() + account.getAddress().getCityName() + account.getAddress().getProvinceName());
        return "success2";
    }
@RequestMapping("/updateAccount")
    public String updateAccount(User user) {
        System.out.println("更新了账户。。。。"+user);
        return "success2";
    }
<form action="account/saveAccount" method="post">
      账户名称:<input type="text" name="name" ><br/>
      账户金额:<input type="text" name="money" ><br/>
      账户省份:<input type="text" name="address.provinceName" ><br/>
      账户城市:<input type="text" name="address.cityName" ><br/>
      <input type="submit" value="保存">
</form>
<form action="account/updateAccount" method="post">
      用户名称:<input type="text" name="username" ><br/>
      用户密码:<input type="password" name="password" ><br/>
      用户年龄:<input type="text" name="age" ><br/>
      账户 1 名称:<input type="text" name="accounts[0].name" ><br/>
      账户 1 金额:<input type="text" name="accounts[0].money" ><br/>
      账户 2 名称:<input type="text" name="accounts[1].name" ><br/>
      账户 2 金额:<input type="text" name="accounts[1].money" ><br/>
      账户 3 名称:<input type="text" name="accountMap['one'].name" ><br/>
      账户 3 金额:<input type="text" name="accountMap['one'].money" ><br/>
      账户 4 名称:<input type="text" name="accountMap['two'].name" ><br/>
      账户 4 金额:<input type="text" name="accountMap['two'].money" ><br/>
      <input type="submit" value="保存">
</form>

请求参数乱码问题

post 请求方式:

在 web.xml 中配置一个过滤器

<!-- 配置 springMVC 编码过滤器 -->
  <filter>
    <filter-name>CharacterEncodingFilter</filter-name>
    <filter-
class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
    <!-- 设置过滤器中的属性值 -->
    <init-param>
      <param-name>encoding</param-name>
      <param-value>UTF-8</param-value>
    </init-param>
    <!-- 启动过滤器 -->
    <init-param>
      <param-name>forceEncoding</param-name>
      <param-value>true</param-value>
    </init-param>
  </filter>
  <!-- 过滤所有请求 -->
  <filter-mapping>
    <filter-name>CharacterEncodingFilter</filter-name>
    <url-pattern>/*</url-pattern>
  </filter-mapping>
get 请求方式:

tomcat 对 GET 和 POST 请求处理方式是不同的,GET 请求的编码问题,要改 tomcat 的 server.xml配
置文件,如下:

<Connector connectionTimeout="20000" port="8080"
protocol="HTTP/1.1" redirectPort="8443"/>
    修改为
 <Connector connectionTimeout="20000" port="8080"
protocol="HTTP/1.1" redirectPort="8443"
useBodyEncodingForURI="true"/>

标签

评论

this is is footer