【SpringMVC】1—SpringMVC基础操作
⭐⭐⭐⭐⭐⭐
Github主页👉https://github.com/A-BigTree
笔记链接👉https://github.com/A-BigTree/Code_Learning
⭐⭐⭐⭐⭐⭐
如果可以,麻烦各位看官顺手点个star~😊
如果文章对你有所帮助,可以点赞👍收藏⭐支持一下博主~😆
文章目录
- 1 SpringMVC基础操作
-
- 1.1 SpringMVC概述
-
- 1.1.1 SpringMVC优势
- 1.1.2 表述层要解决的问题
- 1.1.3 SpringMVC代码对比
-
- 基于原生Servlet API开发
- 基于SpringMVC开发
- 1.2 HelloWorld
-
- 1.2.1 功能需求
-
- 访问首页
- 在首页点超链接
- 1.2.2 搭建环境
-
- 导入依赖
- web.xml
- Spring配置文件
- logback.xml配置文件
- 1.2.3 代码实现
-
- 实现访问首页
- portal.html编写超链接
- 目标页面target.html
- 声明handler方法
- 1.2.4 整套流程解析
- 1.3 `@RequestMapping`注解
-
- 1.3.1 匹配方式说明
-
- 精确匹配
- 模糊匹配
- 1.3.2 在类级别标记
-
- 超链接的HTML标签
- 仅在标记方法上的`@RequestMapping`注解
- 分别标记类和方法上
- 1.3.3 附加请求方式
-
- 请求方式
- 注解附加请求方式
- 进阶版
- 1.4 获取请求参数
-
- 1.4.1 一名一值
-
- 超链接
- `@RequestParam`注解
- 省略注解
- 关闭请求参数必需
- 参数设置默认值
- 1.4.2 一名多值
-
- 表单
- handler方法
- 1.4.3 表单对应模型
-
- 表单
- 处理方法
- POST请求乱码问题
- 1.4.4 表单对应实体类包含级联属性
-
- 实体类
- 表单
- 处理方法
- 1.4.5 要发送的数据是List
-
- 额外封装一层
- 表单
- 处理方法
- 1.5 `@CookieValue`注解
-
- 1.5.1 作用
- 1.5.2 用法
- 1.6 页面跳转控制
-
- 1.6.1 准备工作
-
- 创建范围之外的页面`outter.html`
- 配置SpringMVC配置文件
- 1.6.2 使用指令
-
- 转发指令
- 重定向指令
- 1.6.3 指令处理在源码中的位置
-
- 所在的类
- 所在方法
- 1.7 获取原生Servlet API对象
-
- 1.7.1 原生Servlet API
- 1.7.2 可以直接拿到对象
-
- 创建超链接
- 处理方法
- 1.7.3 获取`ServletContext`
-
- 通过`HttpSession`获取
- 通过IOC容器注入
- 1.7.4 原生对象和IOC容器关系
- 1.8 属性域
-
- 1.8.1 属性域作用
- 1.8.2 请求域操作
-
- `Model`类型形参
- `ModelMap`类型
- `Map`类型的形参
- 原生request对象
- `ModelAndView`对象
- 1.8.3 模型的本质
-
- 关系
- 1.8.4 框架底层将模型存入请求域
- 1.8.5 会话域
- 1.8.6 应用域
- 1.9 静态资源访问
-
- 1.9.1 静态资源的概念
- 1.9.2 SpringMVC环境下的静态资源问题
-
- 斜杠情况
- 扩展名情况
- 1.10 `mvc:view-controller`
-
- 1.10.1 需求情景
- 1.10.2 具体操作
-
- 配置
- 新的问题
- 1.10.3 底层分析
-
- `HabdlerMapping`
- 三个标签都没有配置
- 增加一个标签
- 全部配置三个标签
- 结论
- 1.11 表单标签
-
- 1.11.1 回显简单标签
-
- 创建用于测试的实体类
- 创建handler方法
- `form-simple.html`页面表单回显
- 1.11.2 回显带选择功能的标签
-
- 总体思路
- 创建实体类
- handler方法
- 页面表单回显
1 SpringMVC基础操作
1.1 SpringMVC概述
1.1.1 SpringMVC优势
SpringMVC是Spring为表述层开发提供的一整套完备的解决方案。
在表述层框架历经Strust
、WebWork
、Strust2
等诸多产品的历代更迭之后,目前业界普遍选择了SpringMVC作为Java EE
项目表述层开发的首选方案。之所以能做到这一点,是因为SpringMVC具备如下显著优势:
- Spring 家族原生产品,与IOC容器等基础设施无缝对接;
- 表述层各细分领域需要解决的问题全方位覆盖,提供全面解决方案;
- 代码清新简洁,大幅度提升开发效率;
- 内部组件化程度高,可插拔式组件即插即用,想要什么功能配置相应组件即可;
- 性能卓著,尤其适合现代大型、超大型互联网项目要求;
1.1.2 表述层要解决的问题
- 请求映射
- 数据输入
- 视图界面
- 请求分发
- 表单回显
- 会话控制
- 过滤拦截
- 异步交互
- 文件上传
- 文件下载
- 数据校验
- 类型转换
1.1.3 SpringMVC代码对比
基于原生Servlet API开发
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { String userName = request.getParameter("userName");System.out.println("userName="+userName);}
基于SpringMVC开发
@RequestMapping("/user/login")
public String login(@RequestParam("userName") String userName){log.debug("userName="+userName);return "result";
}
1.2 HelloWorld
1.2.1 功能需求
访问首页
在首页点超链接
1.2.2 搭建环境
导入依赖
<dependencies><!-- SpringMVC --><dependency><groupId>org.springframework</groupId><artifactId>spring-webmvc</artifactId><version>5.3.1</version></dependency><!-- 日志 --><dependency><groupId>ch.qos.logback</groupId><artifactId>logback-classic</artifactId><version>1.2.3</version></dependency><!-- ServletAPI --><dependency><groupId>javax.servlet</groupId><artifactId>javax.servlet-api</artifactId><version>3.1.0</version><scope>provided</scope></dependency><!-- Spring5和Thymeleaf整合包 --><dependency><groupId>org.thymeleaf</groupId><artifactId>thymeleaf-spring5</artifactId><version>3.0.12.RELEASE</version></dependency><!-- Lombok --><dependency><groupId>org.projectlombok</groupId><artifactId>lombok</artifactId><version>1.18.12</version><scope>provided</scope></dependency>
</dependencies>
web.xml
<!-- 配置SpringMVC中负责处理请求的核心Servlet,也被称为SpringMVC的前端控制器 -->
<servlet><servlet-name>DispatcherServlet</servlet-name><!-- DispatcherServlet的全类名 --><servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class><!-- 通过初始化参数指定SpringMVC配置文件位置 --><init-param><!-- 如果不记得contextConfigLocation配置项的名称,可以到DispatcherServlet的父类FrameworkServlet中查找 --><param-name>contextConfigLocation</param-name><!-- 使用classpath:说明这个路径从类路径的根目录开始才查找 --><param-value>classpath:spring-mvc.xml</param-value></init-param><!-- 作为框架的核心组件,在启动过程中有大量的初始化操作要做,这些操作放在第一次请求时才执行非常不恰当 --><!-- 我们应该将DispatcherServlet设置为随Web应用一起启动 --><load-on-startup>1</load-on-startup></servlet><servlet-mapping><servlet-name>DispatcherServlet</servlet-name><!-- 对DispatcherServlet来说,url-pattern有两种方式配置 --><!-- 方式一:配置“/”,表示匹配整个Web应用范围内所有请求。这里有一个硬性规定:不能写成“/*”。只有这一个地方有这个特殊要求,以后我们再配置Filter还是可以正常写“/*”。 --><!-- 方式二:配置“*.扩展名”,表示匹配整个Web应用范围内部分请求 --><url-pattern>/</url-pattern>
</servlet-mapping>
Spring配置文件
<!-- 自动扫描包 -->
<context:component-scan base-package="com.atguigu.mvc.handler"/><!-- Thymeleaf视图解析器 -->
<bean id="viewResolver" class="org.thymeleaf.spring5.view.ThymeleafViewResolver"><property name="order" value="1"/><property name="characterEncoding" value="UTF-8"/><property name="templateEngine"><bean class="org.thymeleaf.spring5.SpringTemplateEngine"><property name="templateResolver"><bean class="org.thymeleaf.spring5.templateresolver.SpringResourceTemplateResolver"><!-- 视图前缀 --><property name="prefix" value="/WEB-INF/templates/"/><!-- 视图后缀 --><property name="suffix" value=".html"/><property name="templateMode" value="HTML5"/><property name="characterEncoding" value="UTF-8" /></bean></property></bean></property>
</bean>
logback.xml配置文件
<?xml version="1.0" encoding="UTF-8"?>
<configuration debug="true"><!-- 指定日志输出的位置,ConsoleAppender表示输出到控制台 --><appender name="STDOUT"class="ch.qos.logback.core.ConsoleAppender"><encoder><!-- 日志输出的格式 --><!-- 按照顺序分别是:时间、日志级别、线程名称、打印日志的类、日志主体内容、换行 --><pattern>[%d{HH:mm:ss.SSS}] [%-5level] [%thread] [%logger] [%msg]%n</pattern><charset>UTF-8</charset></encoder></appender><!-- 设置全局日志级别。日志级别按顺序分别是:TRACE、DEBUG、INFO、WARN、ERROR --><!-- 指定任何一个日志级别都只打印当前级别和后面级别的日志。 --><root level="INFO"><!-- 指定打印日志的appender,这里通过“STDOUT”引用了前面配置的appender --><appender-ref ref="STDOUT" /></root><!-- 根据特殊需求指定局部日志级别,可也是包名或全类名。 --><logger name="com.atguigu" level="DEBUG" /></configuration>
1.2.3 代码实现
实现访问首页
@Slf4j
@Controller
public class Demo01HelloHandler {// @RequestMapping注解在请求地址和Java方法之间建立映射关系@RequestMapping("/")public String showPortal() {return "portal";}}
portal.html编写超链接
<!DOCTYPE html>
<html lang="en" xmlns:th="http://www.thymeleaf.org">
<head><meta charset="UTF-8"><title>首页</title>
</head>
<body><h1>首页</h1><!-- 以后我们会越来越倾向于用一句话来作为请求的URL地址,在这样的一句话中使用“/”分隔各个单词 --><!-- say hello to spring mvc --><!-- /say/hello/to/spring/mvc --><a th:href="@{/say/hello/to/spring/mvc}">HelloWorld</a><br/></body>
</html>
目标页面target.html
<!DOCTYPE html>
<html lang="en" xmlns:th="http://www.thymeleaf.org">
<head><meta charset="UTF-8"><title>目标页面</title>
</head>
<body><h1>目标页面</h1><a th:href="@{/}">回首页</a></body>
</html>
声明handler方法
// 以后我们会越来越倾向于用一句话来作为请求的URL地址
// 在这样的一句话中使用“/”分隔各个单词
@RequestMapping("/say/hello/to/spring/mvc")
public String sayHello() {// 方法内部打印日志,证明 SpringMVC 确实调用了这个方法来处理请求log.debug("我是 SpringMVC 的 Hello world。");return "target";
}
1.2.4 整套流程解析
1.3 @RequestMapping
注解
1.3.1 匹配方式说明
精确匹配
在@RequestMapping
注解指定URL地址时,不使用任何通配符,按照请求地址进行精确匹配。
<a th:href="@{/say/hello/to/spring/mvc}">HelloWorld</a><br/>
@RequestMapping("/say/hello/to/spring/mvc")
模糊匹配
在@RequestMapping
注解指定URL地址时,通过使用通配符,匹配多个类似的地址。
<h3>测试@RequestMapping注解匹配方式</h3>
<a th:href="@{/fruit/apple}">@RequestMapping模糊匹配[apple]</a><br/>
<a th:href="@{/fruit/orange}">@RequestMapping模糊匹配[orange]</a><br/>
<a th:href="@{/fruit/banana}">@RequestMapping模糊匹配[banana]</a><br/>
@RequestMapping("/fruit/*")
单层匹配和多层匹配:
/*
:只能匹配URL地址中的一层,如果想准确匹配两层,那么就写“/*/*
”以此类推;/
:可以匹配URL地址中的多层。
其中所谓的一层或多层是指一个URL地址字符串被/
划分出来的各个层次 这个知识点虽然对于@RequestMapping
注解来说实用性不大,但是将来配置拦截器的时候也遵循这个规则。
1.3.2 在类级别标记
超链接的HTML标签
<h3>测试@RequestMapping注解标记在类上</h3>
<a th:href="@{/user/login}">用户登录</a><br/>
<a th:href="@{/user/register}">用户注册</a><br/>
<a th:href="@{/user/logout}">用户退出</a><br/>
仅在标记方法上的@RequestMapping
注解
@RequestMapping("/user/login")
@RequestMapping("/user/register")
@RequestMapping("/user/logout")
分别标记类和方法上
在类级别:抽取各个方法上@RequestMapping注解地址中前面重复的部分
@RequestMapping("/user")
在方法级别:省略被类级别抽取的部分
@RequestMapping("/login")
@RequestMapping("/register")
@RequestMapping("/logout")
1.3.3 附加请求方式
请求方式
HTTP 协议定义了八种请求方式,在 SpringMVC 中封装到了下面这个枚举类:
public enum RequestMethod {GET, HEAD, POST, PUT, PATCH, DELETE, OPTIONS, TRACE}
注解附加请求方式
<h3>测试@RequestMapping注解限定请求方式</h3>
<a th:href="@{/emp}">同地址GET请求</a><br/>
<form th:action="@{/emp}" method="post"><button type="submit">同地址POST请求</button>
</form>
<br/>
处理 GET 请求:
@RequestMapping(value = "/emp", method = RequestMethod.GET)
public String empGet() {log.debug("GET 请求");return "target";
}
处理 POST 请求:
@RequestMapping(value = "/emp", method = RequestMethod.POST)
public String empPost() {log.debug("POST 请求");return "target";
}
进阶版
原版 | 进阶版 |
---|---|
@RequestMapping(value = “/emp”, method = RequestMethod.GET) |
@GetMapping(“/emp”) |
@RequestMapping(value = “/emp”, method = RequestMethod.POST) |
@PostMapping(“/emp”) |
除了 @GetMapping
、@PostMapping
还有下面几个类似的注解:
@PutMapping
@DeleteMapping
@PatchMapping
另外需要注意:进阶版的这几个注解是从 4.3 版本才开始有,低于 4.3 版本无法使用。
1.4 获取请求参数
1.4.1 一名一值
超链接
<a th:href="@{/param/one/name/one/value(userName='tom')}">一个名字一个值的情况</a><br/>
@RequestParam
注解
@RequestMapping("/param/one/name/one/value")
public String oneNameOneValue(// 使用@RequestParam注解标记handler方法的形参// SpringMVC 会将获取到的请求参数从形参位置给我们传进来@RequestParam("userName") String userName
) {log.debug("获取到请求参数:" + userName);return "target";
}
省略注解
@RequestMapping("/param/one/name/one/value")
public String oneNameOneValue(// 当请求参数名和形参名一致,可以省略@RequestParam("userName")注解// 但是,省略后代码可读性下降而且将来在SpringCloud中不能省略,所以建议还是不要省略String userName
) {logger.debug("★获取到请求参数:" + userName);return "target";
}
关闭请求参数必需
required 属性设置为 false 表示这个请求参数可有可无:
@RequestParam(value = "userName", required = false)
参数设置默认值
使用defaultValue
属性给请求参数设置默认值:
@RequestParam(value = "userName", required = false, defaultValue = "missing")
此时required
属性可以继续保持默认值:
@RequestParam(value = "userName", defaultValue = "missing")
1.4.2 一名多值
表单
<form th:action="@{/param/one/name/multi/value}" method="post">请选择你最喜欢的球队:<input type="checkbox" name="team" value="Brazil"/>巴西<input type="checkbox" name="team" value="German"/>德国<input type="checkbox" name="team" value="French"/>法国<input type="checkbox" name="team" value="Holland"/>荷兰<input type="checkbox" name="team" value="Italian"/>意大利<input type="checkbox" name="team" value="China"/>中国<br/><input type="submit" value="保存"/>
</form>
handler方法
@RequestMapping("/param/one/name/multi/value")
public String oneNameMultiValue(// 在服务器端 handler 方法中,使用一个能够存储多个数据的容器就能接收一个名字对应的多个值请求参数@RequestParam("team") List<String> teamList) {for (String team : teamList) {logger.debug("team = " + team);}return "target";
}
1.4.3 表单对应模型
表单
<form th:action="@{/emp/save}" method="post">姓名:<input type="text" name="empName"/><br/>年龄:<input type="text" name="empAge"/><br/>工资:<input type="text" name="empSalary"/><br/><input type="submit" value="保存"/>
</form>
处理方法
@RequestMapping("/param/form/to/entity")
public String formToEntity(// SpringMVC 会自动调用实体类中的 setXxx() 注入请求参数Employee employee) {logger.debug(employee.toString());return "target";
}
POST请求乱码问题
到 web.xml 中配置 CharacterEncodingFilter 即可:
<!-- 配置过滤器解决 POST 请求的字符乱码问题 -->
<filter><filter-name>CharacterEncodingFilter</filter-name><filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class><!-- encoding参数指定要使用的字符集名称 --><init-param><param-name>encoding</param-name><param-value>UTF-8</param-value></init-param><!-- 请求强制编码 --><init-param><param-name>forceRequestEncoding</param-name><param-value>true</param-value></init-param><!-- 响应强制编码 --><init-param><param-name>forceResponseEncoding</param-name><param-value>true</param-value></init-param>
</filter>
<filter-mapping><filter-name>CharacterEncodingFilter</filter-name><url-pattern>/*</url-pattern>
</filter-mapping>
注1:在较低版本的 SpringMVC 中,
forceRequestEncoding
属性、forceResponseEncoding
属性没有分开,它们是一个forceEncoding
属性。这里需要注意一下;注2:由于
CharacterEncodingFilter
是通过request.setCharacterEncoding(encoding)
来设置请求字符集,所以在此操作前不能有任何的request.getParameter()
操作。在设置字符集之前获取过请求参数,那么设置字符集的操作将无效。
1.4.4 表单对应实体类包含级联属性
实体类
@Data
@NoArgsConstructor
@AllArgsConstructor
public class Student {private String stuName;private School school;private List<Subject> subjectList;}
表单
<!-- 提交数据的表单 -->
<form th:action="@{/param/cascad}" method="post">stuName:<input type="text" name="stuName" value="tom"/><br/>school.schoolName:<input type="text" name="school.schoolName" value="atguigu"/><br/>subjectList[0].subjectName:<input type="text" name="subjectList[0].subjectName" value="java"/><br/>subjectList[1].subjectName:<input type="text" name="subjectList[1].subjectName" value="php"/><br/>subjectList[2].subjectName:<input type="text" name="subjectList[2].subjectName" value="javascript"/><br/>subjectList[3].subjectName:<input type="text" name="subjectList[3].subjectName" value="css"/><br/>subjectList[4].subjectName:<input type="text" name="subjectList[4].subjectName" value="vue"/><br/><input type="submit" value="保存"/>
</form>
处理方法
@RequestMapping("/param/form/to/nested/entity")
public String formToNestedEntity(// SpringMVC 自己懂得注入级联属性,只要属性名和对应的getXxx()、setXxx()匹配即可Student student) {logger.debug(student.toString());return "target";
}
1.4.5 要发送的数据是List
额外封装一层
public class EmployeeParam {private List<Employee> employeeList;……
表单
直接发送 List<Employee>:<br/>
<form th:action="@{/param/list/emp}" method="post">1号员工姓名:<input type="text" name="employeeList[0].empName" /><br/>1号员工年龄:<input type="text" name="employeeList[0].empAge" /><br/>1号员工工资:<input type="text" name="employeeList[0].empSalary" /><br/>2号员工姓名:<input type="text" name="employeeList[1].empName" /><br/>2号员工年龄:<input type="text" name="employeeList[1].empAge" /><br/>2号员工工资:<input type="text" name="employeeList[1].empSalary" /><br/><button type="submit">保存</button>
</form>
处理方法
@RequestMapping("/param/list/emp")
public String saveEmpList(// SpringMVC 访问这里实体类的setEmployeeList()方法注入数据EmployeeParam employeeParam
) {List<Employee> employeeList = employeeParam.getEmployeeList();for (Employee employee : employeeList) {logger.debug(employee.toString());}return "target";
}
1.5 @CookieValue
注解
1.5.1 作用
获取当前请求中的Cookie数据。
1.5.2 用法
@RequestMapping("/request/cookie")
public String getCookie(// 使用 @CookieValue 注解获取指定名称的 Cookie 数据// name 或 value 属性:指定Cookie 名称// defaultValue 属性:设置默认值@CookieValue(value = "JSESSIONID", defaultValue = "missing") String cookieValue,// 形参位置声明 HttpSession 类型的参数即可获取 HttpSession 对象HttpSession session
) {log.debug("cookieValue = " + cookieValue);return "target";
}
1.6 页面跳转控制
1.6.1 准备工作
- 准备一个地址在前后缀范围之外的页面;
- 让这个页面能够成功访问;
创建范围之外的页面outter.html
<body><h1>范围之外页面</h1></body>
很多时候我们创建静态资源文件时,IDEA并不会自动帮我们放在构建目录下,进而就会导致部署到服务器上运行的时候没有这个文件,此时需要我们自己手动构建。
配置SpringMVC配置文件
<mvc:annotation-driven/>
<mvc:default-servlet-handler/>
1.6.2 使用指令
转发指令
@RequestMapping("/test/forward/command")
public String forwardCommand() {// 需求:要转发前往的目标地址不在视图前缀指定的范围内,// 通过返回逻辑视图、拼接前缀后缀得到的物理视图无法达到目标地址// 转发到指定的地址:return "forward:/outter.html";
}
重定向指令
@RequestMapping("/test/redirect/command")
public String redirectCommand() {// 重定向到指定的地址:// 这个地址由 SpringMVC 框架负责在前面附加 contextPath,所以我们不能加,我们加了就加多了// 框架增加 contextPath 后:/demo/outter.html// 我们多加一个:/demo/demo/outter.htmlreturn "redirect:/outter.html";
}
1.6.3 指令处理在源码中的位置
所在的类
org.thymeleaf.spring5.view.ThymeleafViewResolver
所在方法
1.7 获取原生Servlet API对象
1.7.1 原生Servlet API
HttpServletRequest
HttpServletResponse
HttpSession
ServletContext
原生:最原始的、本真的,没有经过任何的加工、包装和处理。
API:直接翻译过来是应用程序接口的意思。对我们来说,提到 API 这个词的时候,通常指的是在某个特定的领域,已经封装好可以直接使用的一套技术体系。很多时候,特定领域的技术规范都是对外暴露一组接口作为这个领域的技术标准,然后又在这个标准下有具体实现。
1.7.2 可以直接拿到对象
创建超链接
<a th:href="@{/original/api/direct}">可以直接得到的三个</a><br/>
处理方法
@RequestMapping("/original/api/direct")
public String getOriginalAPIDirect(// 有需要使用的 Servlet API 直接在形参位置声明即可。// 需要使用就写上,不用就不写,开发体验很好,这里给 SpringMVC 点赞HttpServletRequest request,HttpServletResponse response,HttpSession session
) {logger.debug(request.toString());logger.debug(response.toString());logger.debug(session.toString());return "target";
}
ServletContext
对象没法通过形参声明的方式直接获取,如果非要在形参位置声明ServletContext
类型的变量,那么会抛出异常。
1.7.3 获取ServletContext
通过HttpSession
获取
@RequestMapping("/original/servlet/context/first/way")
public String originalServletContextFirstWay(HttpSession session) {// 获取ServletContext对象的方法一:通过HttpSession对象获取ServletContext servletContext = session.getServletContext();logger.debug(servletContext.toString());return "target";
}
通过IOC容器注入
// 获取ServletContext对象的方法二:从 IOC 容器中直接注入
@Autowired
private ServletContext servletContext;@RequestMapping("/original/servlet/context/second/way")
public String originalServletContextSecondWay() {logger.debug(this.servletContext.toString());return "target";
}
1.7.4 原生对象和IOC容器关系
1.8 属性域
1.8.1 属性域作用
1.8.2 请求域操作
请求域是实际开发中使用最多的属性域,所以SpringMVC也提供了多种不同方式来操作:
Model
类型形参
@RequestMapping("/attr/request/model")
public String testAttrRequestModel(// 在形参位置声明Model类型变量,用于存储模型数据Model model) {// 我们将数据存入模型,SpringMVC 会帮我们把模型数据存入请求域// 存入请求域这个动作也被称为暴露到请求域model.addAttribute("requestScopeMessageModel","i am very happy[model]");return "target";
}
ModelMap
类型
@RequestMapping("/attr/request/model/map")
public String testAttrRequestModelMap(// 在形参位置声明ModelMap类型变量,用于存储模型数据ModelMap modelMap) {// 我们将数据存入模型,SpringMVC 会帮我们把模型数据存入请求域// 存入请求域这个动作也被称为暴露到请求域modelMap.addAttribute("requestScopeMessageModelMap","i am very happy[model map]");return "target";
}
Map
类型的形参
@RequestMapping("/attr/request/map")
public String testAttrRequestMap(// 在形参位置声明Map类型变量,用于存储模型数据Map<String, Object> map) {// 我们将数据存入模型,SpringMVC 会帮我们把模型数据存入请求域// 存入请求域这个动作也被称为暴露到请求域map.put("requestScopeMessageMap", "i am very happy[map]");return "target";
}
原生request对象
@RequestMapping("/attr/request/original")
public String testAttrOriginalRequest(// 拿到原生对象,就可以调用原生方法执行各种操作HttpServletRequest request) {request.setAttribute("requestScopeMessageOriginal", "i am very happy[original]");return "target";
}
ModelAndView
对象
@RequestMapping("/attr/request/mav")
public ModelAndView testAttrByModelAndView() {// 1.创建ModelAndView对象ModelAndView modelAndView = new ModelAndView();// 2.存入模型数据modelAndView.addObject("requestScopeMessageMAV", "i am very happy[mav]");// 3.设置视图名称modelAndView.setViewName("target");return modelAndView;
}
1.8.3 模型的本质
SpringMVC 传入的Model
、ModelMap
、Map
类型的参数其实本质上都是BindingAwareModelMap
类型的。
关系
1.8.4 框架底层将模型存入请求域
所在类:org.thymeleaf.context.WebEngineContext
内部类RequestAttributesVariablesMap
所在方法:setVariable(String name, Obiect value)
;
1.8.5 会话域
使用会话域最简单直接的办法就是使用原生的HttpSession
对象。
@RequestMapping("/attr/session")
public String attrSession(// 使用会话域最简单直接的办法就是使用原生的 HttpSession 对象HttpSession session) {session.setAttribute("sessionScopeMessage", "i am haha ...");return "target";
}
1.8.6 应用域
@Autowired
private ServletContext servletContext;@RequestMapping("/attr/application")
public String attrApplication() {servletContext.setAttribute("appScopeMsg", "i am hungry...");return "target";
}
1.9 静态资源访问
1.9.1 静态资源的概念
资源本身已经是可以直接拿到浏览器上使用的程度了,不需要在服务器端做任何运算、处理。典型的静态资源包括:
- 纯HTML文件
- 图片
- CSS文件
- JavaScript文件
- ……
1.9.2 SpringMVC环境下的静态资源问题
斜杠情况
DispatcherServlet
的url-pattern
标签配置的是/
。意味着整个Web应用范围内所有请求都由SpringMVC来处理。
- 对SpringMVC来说,必须有对应的
@RequestMapping
才能找到处理请求的方法; - 现在
images/mi.jpg
请求没有对应的@RequestMapping
所以返回404;
解决办法:
<!-- 加入这个配置,SpringMVC 就会在遇到没有 @RequestMapping 的请求时放它过去 -->
<!-- 所谓放它过去就是让这个请求去找它原本要访问的资源 -->
<mvc:default-servlet-handler/>
- 新的问题:其他原本正常的请求访问不了了
- 进一步解决问题:再增加一个配置
<!-- 开启SpringMVC的注解驱动功能。这个配置也被称为SpringMVC的标配。 -->
<!-- 标配:因为 SpringMVC 环境下非常多的功能都要求必须打开注解驱动才能正常工作。 -->
<mvc:annotation-driven/>
大体机制:SpringMVC首先查找是否存在和当前请求对应的@RequestMapping
;如果没有,则调用handleRequest()
方法转发到目标资源。
@Override
public void handleRequest(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {Assert.state(this.servletContext != null, "No ServletContext set");RequestDispatcher rd = this.servletContext.getNamedDispatcher(this.defaultServletName);if (rd == null) {throw new IllegalStateException("A RequestDispatcher could not be located for the default servlet '" +this.defaultServletName + "'");}// 这里执行请求转发操作rd.forward(request, response);
}
扩展名情况
修改url-pattern
:
<servlet><servlet-name>DispatcherServlet</servlet-name><servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class><init-param><param-name>contextConfigLocation</param-name><param-value>classpath:spring-mvc.xml</param-value></init-param><load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping><servlet-name>DispatcherServlet</servlet-name><!--<url-pattern>/</url-pattern>--><!-- 以扩展名方式匹配 SpringMVC 要处理的请求 --><!-- 此时要求请求扩展名必须是 html,SpringMVC 才会处理这个请求 --><url-pattern>*.html</url-pattern>
</servlet-mapping>
效果:
- 图片直接就可以访问了。因为请求扩展名不是
html
,不会受到 SpringMVC影响。 - 其他请求:做下面两个操作才可以正常访问
- 需要在超链接地址后面附加
html
扩展名; - 在
@RequestMapping
注解指定的URL地址中也附加html
扩展名;
- 需要在超链接地址后面附加
特殊情况:在请求扩展名就是html
的情况下,访问HTML静态页面。仍然需要开启mvc:default-servlet-handler
配置。
原因是:扩展名为html
会让 SpringMVC 觉得这个请求归它管,它一看没有@RequestMapping
就会返回 404。而打开了mvc:default-servlet-handler
配置就会转发到 HTML 页面。
1.10 mvc:view-controller
1.10.1 需求情景
在一个handler方法中,仅仅只是完成@RequestMapping
映射,将请求转发到目标视图,除此之外没有任何其他代码。此时可以使用 SpringMVC 配置文件中的配置代替这样的handler方法。
1.10.2 具体操作
配置
在SpringMVC配置文件中使用mvc:view-controller
配置:
<mvc:view-controller path="/index.html" view-name="portal"/>
同时,handler类中就可以去掉被代替的方法。
新的问题
加入mvc:view-controller
配置后,其他正常@RequestMapping
将失效。此时还是需要加入mvc:annotation-driven
来解决。
1.10.3 底层分析
HabdlerMapping
见名知意,HandlerMapping
封装的数据包含了请求地址和handler
方法之间的映射关系。所以请求访问是否能生效关键要看HandlerMapping
在IOC容器中加载的情况。为了看到这一点,我们可以在DispatcherServlet
中找到doDispatch()
方法设置断点。之所以选择这个方法,是因为每一个由SpringMVC处理的请求都会经过这里,便于操作。
三个标签都没有配置
SpringMVC加载了三个HandlerMapping
:
org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping
;org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping
;org.springframework.web.servlet.function.support.RouterFunctionMapping
;
其中RequestMappingHandlerMapping
封装了@RequestMapping
相关请求,有它在@RequestMapping
相关请求就能访问到。
增加一个标签
配置了mvc:view-controller
或mvc:default-servlet-handler
之后。
SpringMVC加载了两个HandlerMapping
:
org.springframework.web.servlet.handler.SimpleUrlHandlerMapping
;org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping
;
全部配置三个标签
配置全部mvc:view-controller
、mvc:default-servlet-handler
、mvc:annotation-driven
三个标签。
SpringMVC加载了略有不同的三个HandlerMapping
:
org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping
;org.springframework.web.servlet.handler.SimpleUrlHandlerMapping
;org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping
;
结论
在配置不同的情况下,SpringMVC底层加载的组件不同,特定功能需要特定组件的支持。当特定功能所需组件没有加入到IOC容器中的时候,对应的功能就无法使用了。
还有一点是:mvc:annotation-driven
是SpringMVC标配,必加。
1.11 表单标签
表单标签主要的目的是在页面上实现表单回显。最典型的情况是在修改数据时,把之前旧的数据重新显示出来供用户参考。
1.11.1 回显简单标签
创建用于测试的实体类
@Data
@NoArgsConstructor
@AllArgsConstructor
public class Tiger {private Integer tigerId;private String tigerName;private Double tigerSalary;}
创建handler方法
@RequestMapping("/form/redisplay/simple")
public String simpleTagRedisplay(Model model) {// 1.准备好用来回显表单的实体类对象// 在实际功能中,这里的对象应该是从数据库查询得到Tiger tiger = new Tiger();tiger.setTigerId(5);tiger.setTigerName("tomCat");tiger.setTigerSalary(666.66);// 2.将实体类数据存入模型model.addAttribute("tiger", tiger);return "form-simple";
}
form-simple.html
页面表单回显
<h3>回显Tiger数据</h3><form th:action="@{/save/tiger}" method="post"><!-- th:value 和 th:field 属性都可以 -->老虎的id:<input type="text" name="tigerId" th:value="${tiger.tigerId}" /><br/>老虎的名字:<input type="text" name="tigerName" th:field="${tiger.tigerName}" /><br/>老虎的工资:<input type="text" name="tigerSalary" th:field="${tiger.tigerSalary}" /><br/><button type="submit">保证</button>
</form>
1.11.2 回显带选择功能的标签
总体思路
- 显示标签本身,需要用到一个集合对象来存储标签本身所需要的数据;
- 对标签执行回显操作,需要用到另外的一个实体类;
创建实体类
@Data
@NoArgsConstructor
@AllArgsConstructor
public class Season {// 提交给服务器的值private String submitValue;// 给用户看的值private String showForUserValue;}
@Data
@NoArgsConstructor
@AllArgsConstructor
public class Paige {private Integer paigeId;private String paigeName;private Season season;}
handler方法
@RequestMapping("/form/redisplay/choose")
public String chooseTagRedisplay(Model model) {// 1.准备用来显示标签的数据List<Season> seasonList = new ArrayList<>();seasonList.add(new Season("spring", "春天"));seasonList.add(new Season("summer", "夏天"));seasonList.add(new Season("autumn", "秋天"));seasonList.add(new Season("winter", "冬天"));model.addAttribute("seasonList", seasonList);// 2.准备用来回显表单的实体类数据Paige paige = new Paige();paige.setPaigeId(6);paige.setPaigeName("pig");paige.setSeason(new Season("summer", "夏天"));model.addAttribute("paige", paige);return "form-choose";
}
页面表单回显
单选按钮:
<!-- th:each属性:指定用来生成这一组标签的集合数据 -->
<!-- th:value属性:获取数据用来设置HTML标签的value属性,成为将来提交给服务器的值 -->
<!-- th:text属性:获取数据用来设置HTML标签旁边给用户看的名字 -->
<!-- th:checked属性:判断是否回显(把适合的标签设置为默认被选中) -->
<input type="radio" name="season.submitValue"th:each="season : ${seasonList}"th:value="${season.submitValue}"th:text="${season.showForUserValue}"th:checked="${season.submitValue == paige.season.submitValue}"
/>
下拉列表:
<select name="season.submitValue"><option th:each="season : ${seasonList}"th:value="${season.submitValue}"th:text="${season.showForUserValue}"th:selected="${season.submitValue == paige.season.submitValue}"/>
</select>
多选框:
@RequestMapping("/form/redisplay/choose/multi")
public String chooseMulti(Model model) {// 1.准备用来显示标签的数据List<Season> seasonList = new ArrayList<>();seasonList.add(new Season("spring", "春天"));seasonList.add(new Season("summer", "夏天"));seasonList.add(new Season("autumn", "秋天"));seasonList.add(new Season("winter", "冬天"));model.addAttribute("seasonList", seasonList);// 2.准备用来回显表单的实体类数据List<Season> seasonListForRedisplay = new ArrayList<>();seasonListForRedisplay.add(new Season("summer", "夏天"));seasonListForRedisplay.add(new Season("winter", "冬天"));model.addAttribute("seasonListForRedisplay", seasonListForRedisplay);return "form-multi";
}
- 页面标签:
<!--seasonListForRedisplay.contains(season)用包含回显数据的集合调用contains()方法判断是否应该被选中;传入contains()方法的是生成具体每一个标签时遍历得到的对象-->
<input type="checkbox" name="xxx"th:each="season : ${seasonList}"th:value="${season.submitValue}"th:text="${season.showForUserValue}"th:checked="${seasonListForRedisplay.contains(season)}"
/>
注意:此时需要判断一个对象是否在集合中,那就需要重写equals()
和hashCode()
方法。