一、概述
1、MVC思想
模型-视图-控制器(MVC)是一个众所周知的以设计界面应用程序为基础的设计思想。它主要通过分离模型、视图及控制器在应用程序中的角色将业务逻辑从界面中解耦。通常,模型负责封装应用程序数据在视图层展示。视图仅仅只是展示这些数据,不包含任何业务逻辑。控制器负责接收来自用户的请求,并调用后台服务(service 或者 dao)来处理业务逻辑。处理后,后台业务层可能会返回了一些数据在视图层展示。控制器收集这些数据及准备模型在视图层展示。MVC 模式的核心思想是将业务逻辑从界面中分离出来,允许它们单独改变而不会相互影响。
image.png
2、常见MVC框架比较
运行性能上:
Jsp+servlet>struts1>spring mvc>struts2+freemarker>struts2,ognl, 值栈。
开发效率上,基本正好相反。值得强调的是,spring mvc 开发效率和
struts2 不相上下,但从目前来看,spring mvc 的流行度已远远超过 struts2。
Struts2 的性能低的原因是因为 OGNL(一种表达式语言,通过它简单一致的表达式语法,可以存取对象的任意属性,调用对象的方法,结合 struts2 框架使用)和值栈(简单理解为存放 struts2 action 的堆栈)造成的。所以,如果系统并发量高,可以使用 freemaker 进行显示,而不是采用 OGNL 和值栈。这样,在性能上会有相当大得提高。
3、Spring MVC
Spring MVC 是 Spring 家族中的一个 web 成员, 它是一种基于 Java 的实现了 Web MVC 设计思想的请求驱动类型的轻量级 Web 框架,即使用了 MVC 架构模式的思想,将 web 层进行职责解耦,基于请求驱动指的就是使用请求-响应模型,框架的目的就是帮助我们简化开发,Spring MVC 也是要简化我们日常Web 开发的。
Spring MVC 是服务到工作者思想的实现。前端控制器是DispatcherServlet;应用控制器拆为处理器映射器(Handler Mapping)进行处理器管理和视图解析器(View Resolver)进行视图管理;支持本地化/国际化(Locale)解析及文件上传等;提供了非常灵活的数据验证、格式化和数据绑定机制;提供了强大的约定大于配置(惯例优先原则)的契约式编程支持。
4、Spring能做什么
- 让我们能非常简单的设计出干净的 Web 层;
- 进行更简洁的 Web 层的开发;
- 天生与 Spring 框架集成(如 IoC 容器、AOP 等);
- 提供强大的约定大于配置的契约式编程支持;
- 能简单的进行 Web 层的单元测试;
- 支持灵活的 URL 到页面控制器的映射;
- 非常容易与其他视图技术集成,如 Velocity、FreeMarker 等等,因为模型
数据不放在特定的 API 里,而是放在一个 Model 里(Map 数据结构实现,因
此很容易被其他框架使用); - 非常灵活的数据验证、格式化和数据绑定机制,能使用任何对象进行数据
绑定,不必实现特定框架的 API; - 支持灵活的本地化等解析;
- 更加简单的异常处理;
- 对静态资源的支持;
- 支持 Restful 风格。
二、Spring MVC架构
1、Spring MVC 请求处理流程分析
Spring MVC 框架也是一个基于请求驱动的 Web 框架,并且使用了前端控制
器模式(是用来提供一个集中的请求处理机制,所有的请求都将由一个单一的
处理程序处理来进行设计,再根据请求映射规则分发给相应的页面控制器(动
作/处理器)进行处理。首先让我们整体看一下 Spring MVC 处理请求的流程:
具体执行步骤如下:
1、 首先用户发送请求,请求被 SpringMvc 前端控制器(DispatherServlet)捕获;
2、 前端控制器(DispatherServlet)对请求 URL 解析获取请求 URI,根据 URI, 调用 HandlerMapping;
3、 前端控制器(DispatherServlet)获得返回的 HandlerExecutionChain(包括Handler 对象以及 Handler 对象对应的拦截器);
4、 DispatcherServlet 根据获得的 HandlerExecutionChain,选择一个合适的HandlerAdapter。(附注:如果成功获得 HandlerAdapter 后,此时将开始执行拦截器的 preHandler(...)方法);
5、 HandlerAdapter 根据请求的 Handler 适配并执行对应的Handler;HandlerAdapter(提取 Request 中的模型数据,填充 Handler 入参,开始执行 Handler(Controller)。 在填充 Handler 的入参过程中,根据配置,Spring 将做一些额外的工作:
HttpMessageConveter: 将请求消息(如 Json、xml 等数据)转换成一个对象,将对象转换为指定的响应信息。
数据转换:对请求消息进行数据转换。如 String 转换成 Integer、Double等
数据格式化:对请求消息进行数据格式化。 如将字符串转换成格式化数字或格式化日期等
数据验证: 验证数据的有效性(长度、格式等),验证结果存储到
BindingResult 或 Error 中)
6、 Handler 执行完毕,返回一个 ModelAndView(即模型和视图)给
HandlerAdaptor
7、 HandlerAdaptor 适配器将执行结果 ModelAndView 返回给前端控制器。
8、 前端控制器接收到 ModelAndView 后,请求对应的视图解析器。
9、 视图解析器解析 ModelAndView 后返回对应 View;
10、渲染视图并返回渲染后的视图给前端控制器。
11、最终前端控制器将渲染后的页面响应给用户或客户端
2、Spring MVC核心架构图
3、Spring MVC优势
1、清晰的角色划分:前端控制器(DispatcherServlet)、请求到处理器映射(HandlerMapping)、处理器适配器(HandlerAdapter)、视图解析器(ViewResolver)、处理器或页面控制器(Controller)、验证器( Validator)、命令对象(Command 请求参数绑定到的对象就叫命令对象)、表单对象(Form Object 提供给表单展示和提交到的对象就叫表单对象)。
2、分工明确,而且扩展点相当灵活,可以很容易扩展,虽然几乎不需要;
3、和 Spring 其他框架无缝集成,是其它 Web 框架所不具备的;
4、可适配,通过 HandlerAdapter 可以支持任意的类作为处理器;
5、可定制性,HandlerMapping、ViewResolver 等能够非常简单的定制;
6、功能强大的数据验证、格式化、绑定机制;
7、利用 Spring 提供的 Mock 对象能够非常简单的进行 Web 层单元测试;
8、本地化、主题的解析的支持,使我们更容易进行国际化和主题的切换。
9、强大的 JSP 标签库,使 JSP 编写更容易。还有比如 RESTful(一种软件架构风格,设计风格而不是标准,只是提供了一组设计原则和约束条件。它主要用于客户端和服务器交互类的软件。)风格的支持、简单的文件上传、约定大于配置的契约式编程支持、基于
注解的零配置支持等等。
三、Spring MVC环境搭建
1、新建项目
2、Spring MVC环境 jar包依赖
<?xml version="1.0" encoding="UTF-8"?>
<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/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>com.shsxt</groupId>
<artifactId>springmvc01</artifactId>
<version>1.0-SNAPSHOT</version>
<packaging>war</packaging>
<name>springmvc01 Maven Webapp</name>
<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>
</properties>
<dependencies>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.12</version>
<scope>test</scope>
</dependency>
<!-- spring web -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-web</artifactId>
<version>4.3.2.RELEASE</version>
</dependency>
<!-- spring mvc -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-webmvc</artifactId>
<version>4.3.2.RELEASE</version>
</dependency>
<!-- web servlet -->
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>javax.servlet-api</artifactId>
<version>3.0.1</version>
</dependency>
</dependencies>
<!-- jetty 插件 -->
<build>
<finalName>springmvc01</finalName>
<plugins>
<plugin>
<groupId>org.mortbay.jetty</groupId>
<artifactId>maven-jetty-plugin</artifactId>
<version>6.1.25</version>
<configuration>
<scanIntervalSeconds>10</scanIntervalSeconds>
<contextPath>/springmvc01</contextPath>
</configuration>
</plugin>
</plugins>
</build>
</project>
3、配置 web.xml (前端控制器配置)
<?xml version="1.0" encoding="UTF-8"?>
<web-app id="WebApp_ID" version="3.0"
xmlns="http://java.sun.com/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee
http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd">
<!-- 表示容器启动时 加载上下文配置 这里指定spring 相关配置 -->
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:*.xml</param-value>
</context-param>
<!-- 启用spring容器环境上下文监听 -->
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
<!-- 编码过滤 utf-8 -->
<filter>
<description>char encoding filter</description>
<filter-name>encodingFilter</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>
</filter>
<filter-mapping>
<filter-name>encodingFilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
<!-- servlet请求分发器 -->
<servlet>
<servlet-name>springMvc</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:servlet-context.xml</param-value>
</init-param>
<!-- 表示启动容器时初始化该Servlet -->
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>springMvc</servlet-name>
<!-- 这是拦截请求, /代表拦截所有请求 -->
<url-pattern>/</url-pattern>
</servlet-mapping>
</web-app>
4、servlet-context.xml 配置
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:mvc="http://www.springframework.org/schema/mvc"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:aop="http://www.springframework.org/schema/aop"
xsi:schemaLocation="
http://www.springframework.org/schema/mvc
http://www.springframework.org/schema/mvc/spring-mvc.xsd
http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop.xsd">
<!-- 扫描com.shsxt.controller 下包 -->
<context:component-scan base-package="com.shsxt.controller" />
<!-- mvc 请求映射 处理器与适配器配置-->
<mvc:annotation-driven/>
<!--配置视图解析器 默认的视图解析器- -->
<bean id="defaultViewResolver"
class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<property name="viewClass" value="org.springframework.web.servlet.view.JstlView" />
<property name="contentType" value="text/html" />
<property name="prefix" value="/WEB-INF/jsp/" />
<property name="suffix" value=".jsp" />
</bean>
</beans>
5、页面控制器的编写
/**
* 采用注解扫描形式
*/
@Controller
public class HelloController {
/**
* 请求映射地址 /hello.do
* @return
*/
@RequestMapping("/hello")
public ModelAndView hello(){
ModelAndView mv=new ModelAndView();
mv.addObject("hello", "hello spring mvc");
mv.setViewName("hello");
return mv;
}
}
6、添加视图页面
在 WEB-INF 下新建 jsp 文件夹 ,并在文件加下新建 hello.jsp
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>Title</title>
</head>
<body>
<h3> Hello SpringMVC!!</h3>
</body>
</html>
7、启动 jetty 服务器
即可通过浏览器访问 http://localhost:8080/springmvc01/hello
四、Spring MVC注解特性
1、@Controller 控制器定义
在 spring 3.0 中,通过@controller 标注即可将 class 定义为一个 controller 类。为使 spring 能找到定义为 controller 的 bean,需要在 spring-context 配置文件中增加如下定义:
<context:component-scan base-package="com.shsxt.controller"/>
2、.@RequestMapping
在类前面定义,则将 url 和类绑定。
在方法前面定义,则将 url 和类的方法绑定
@Controller
@RequestMapping("data")
public class DataBindController {
// 简化版
// /hello
@RequestMapping("hello")
public String hello() {
return "hello";
}
// 完整版
@RequestMapping("hello02")
public ModelAndView hello02() {
ModelAndView mv = new ModelAndView();
mv.setViewName("hello");// 设置页面
mv.addObject("a", "shanghai");// 设置响应参数
return mv;
}
}
3、参数绑定
a)基本数据类型、字符串类型绑定
@RequestMapping("hello04")
public String hello04(Integer a, String b) {
System.out.println("a: " + a);
System.out.println("b: " + b);
return "hello";
}
@RequestMapping("hello08")
public String hello08(
@RequestParam(defaultValue = "1") Integer pageNum,
@RequestParam(defaultValue = "10") Integer pageSize) {
System.out.println(pageNum +" = " + pageSize);
return "hello";
}
b)数组类型
@RequestMapping("hello05")
public String hello05(Integer[] ids) {
for (Integer id : ids) {
System.out.println(id);
}
return "hello";
}
c)vo 类型
@RequestMapping("hello07")
public String hello07(User user) {
System.out.println(user);
return "hello";
}
d)List类型
此时 user 实体需要定义 list 属性
public class User {
private Integer id;
private String name;
private Integer age;
private List<Phone> phones = new ArrayList<>();
private Map<String, Object> map = new HashMap();
private Phone phone;
......
}
此时请求格式:
e)map类型
请求格式:
4、向前台传输参数
package com.shsxt.controller;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.SessionAttributes;
import org.springframework.web.servlet.ModelAndView;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpSession;
import java.util.Map;
/**
* Created by xlf on 2019/7/11.
*/
@Controller
@RequestMapping("params")
@SessionAttributes({"userName"})// userName 放入session
public class ParamsController {
// 传统方式
// 放入作用域
@RequestMapping("hello1")
public String hello1(HttpServletRequest request){
request.setAttribute("a", "123");
return "hello";
}
@RequestMapping("hello2")
public String hello2(HttpServletRequest request){
request.getSession().setAttribute("a", "456");
return "hello";
}
@RequestMapping("hello3")
public ModelAndView hello3(){
ModelAndView mv = new ModelAndView();
mv.addObject("a","spring mvc");
mv.addObject("userName","spring boot");
mv.setViewName("hello");
return mv;
}
@RequestMapping("hello4")
public String hello4(Model model){
model.addAttribute("a", "model");
return "hello";
}
@RequestMapping("hello5")
public String hello5(Map map){
map.put("a", "map");
return "hello";
}
@RequestMapping("hello6")
public String hello6(HttpSession session){
session.setAttribute("a", "session");
return "hello";
}
}
5、@SessionAttributes
用于声明 session 级别存储的属性,放置在处理器类上
@Controller
@SessionAttributes({"userName"})// userName 放入 session
public class UserController {
@RequestMapping("/queryUser")
public ModelAndView queryUser(String userName){
ModelAndView mv=new ModelAndView();
mv.addObject("userName", userName);
mv.setViewName("user");
return mv;
}
}
五、请求转发与重定向的问题
@Controller
public class PageController {
// /page01
@RequestMapping("page01")
public String page01(){
return "redirect:v1.jsp";
}
@RequestMapping("page02")
public String page02(){
return "redirect:v1.jsp?a=1&b=2";
}
@RequestMapping("page03")
public String page03(){
return "redirect:v1.jsp?a=1&b=上海";
}
@RequestMapping("page04")
public String page04(RedirectAttributes attr){
attr.addAttribute("a", 1);
attr.addAttribute("b", "上海");
return "redirect:v1.jsp";
}
@RequestMapping("page05")
public String page05(){
System.out.println("page05");
return "redirect:page06";
}
@RequestMapping("page06")
public String page06(){
System.out.println("page06");
return "hello";
}
@RequestMapping("page07")
public String page07(){
System.out.println("page07");
return "forward:page08";
}
@RequestMapping("page08")
public String page08(){
System.out.println("page08");
return "hello";
}
}
六、Spring MVC之Json数据开发
@ResponseBody
该注解用于将 Controller 的方法返回的对象,通过适当的 HttpMessageConverter
转换为指定格式后,写入到 Response 对象的 body 数据区。
返回的数据不是 html 标签的页面,而是其他某种格式的数据时(如 json、xml 等)
使用(通常用于 ajax 请求)
1、添加 json 依赖 jar 包
<!-- 添加 json 依赖 jar 包 -->
<dependency>
<groupId>com.fasterxml.jackson.core</groupId>
<artifactId>jackson-core</artifactId>
<version>2.7.0</version>
</dependency>
<dependency>
<groupId>com.fasterxml.jackson.core</groupId>
<artifactId>jackson-databind</artifactId>
<version>2.7.0</version>
</dependency>
<dependency>
<groupId>com.fasterxml.jackson.core</groupId>
<artifactId>jackson-annotations</artifactId>
<version>2.7.0</version>
</dependency>
2、修改 servlet-context.xml
添加 json 转换器配置
<!-- json 支持 -->
<bean
class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingH
andlerMapping">
</bean>
<bean
class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingH
andlerAdapter">
<property name="messageConverters">
<list>
<bean
class="org.springframework.http.converter.json.MappingJackson2HttpMessageCon
verter" />
</list>
</property>
</bean>
3、Json数据操作
@RequestMapping("/getUser")
@ResponseBody
public User getUser(User user){
System.out.println(user);
return user;
}
七、拦截器
方式一:
实现接口:org.springframework.web.servlet.HandlerInterceptor
/**
* Created by xlf on 2019/7/12.
*/
public class MyHandlerInterceptor implements HandlerInterceptor {
@Override
public boolean preHandle(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o) throws Exception {
System.out.println("before target method...");
/***
* 判断session是否为空
* */
User user = (User) httpServletRequest.getSession().getAttribute("user");
if(null==user){
System.err.println("没有登陆");
return false;
}
return true;// 返回false 整个请求停止; true代表继续执行
}
@Override
public void postHandle(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o, ModelAndView modelAndView) throws Exception {
System.out.println("after target method...");
}
@Override
public void afterCompletion(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o, Exception e) throws Exception {
System.out.println("after view method...");
}
}
方式二:
继承适配器:
/**
* Created by xlf on 2019/7/12.
*/
public class MyHandlerInterceptor2 extends HandlerInterceptorAdapter {
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
System.out.println("preHandle 222");
return true;
}
@Override
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
System.out.println("postHandle 222");
}
@Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
System.out.println("afterCompletion 222");
}
}
配置方式一:
<mvc:interceptors>
<!-- 使用 bean 定义一个 Interceptor
直接定义在 mvc:interceptors 根下面的 Interceptor 将拦截所有的请求 -->
<bean class="com.shsxt.interceptors.MyInterceptor" />
</mvc:interceptors>
配置方式二:
<mvc:interceptors>
<!-- 定义在 mvc:interceptor 下面 拦截所有 test 地址开头的请求-->
<mvc:interceptor>
<mvc:mapping path="/test/*.do" />
<bean class="com.shsxt.interceptors.MyInterceptor" />
</mvc:interceptor>
</mvc:interceptors>
多个拦截器配置(多个拦截器组成一个拦截器链 ,栈式结构 123 321 退出)
<mvc:interceptors>
<mvc:interceptor>
<!-- 拦截所有请求 -->
<mvc:mapping path="/**" />
<bean class="com.shsxt.interceptors.MyInterceptor" />
</mvc:interceptor>
<mvc:interceptor>
<!-- 拦截所有请求 -->
<mvc:mapping path="/**" />
<bean class="com.shsxt.interceptors.MyInterceptor2" />
</mvc:interceptor>
</mvc:interceptors>
八、SpringMVC文件上传
1、Pom 文件修改 添加 commons-fileupload 依赖
<dependency>
<groupId>commons-fileupload</groupId>
<artifactId>commons-fileupload</artifactId>
<version>1.3.2</version>
</dependency>
2、servlet-context.xml配置
<bean id="multipartResolver"
class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
<property name="maxUploadSize">
<value>104857600</value>
</property>
<property name="maxInMemorySize">
<value>4096</value>
</property>
</bean>
3、表单
<form action="uploadFile.do" method="post" enctype="multipart/form�data">
<input type="file" name="file" />
<button type="submit"> 提交</button>
</form>
4、controller层
@Controller
public class FileController {
@RequestMapping("uploadFile")
@ResponseBody
public String uploadFile(HttpServletRequest request){
// 1. 强制转换
MultipartHttpServletRequest mr = (MultipartHttpServletRequest) request;
// 2. 获取上传文件
MultipartFile file = mr.getFile("file");
// 3. 非空判断
if(null!=file && !file.isEmpty()){
// 获取上传文件夹的路径
String path=request.getSession().getServletContext().getRealPath("upload");
// 获取原始名字
String filename = file.getOriginalFilename();
// 4. 存储
try {
file.transferTo(new File(path, filename));
} catch (IOException e) {
e.printStackTrace();
}
}
return "success";
}