Spring Web MVC项目的创建及使用
一、什么是Spring Web MVC?
Spring Web MVC 是基于 Servlet API 构建的原始 Web 框架,从⼀开始就包含在 Spring 框架中,通常被称为Spring MVC。
1.1 MVC的定义
MVC 是 Model View Controller 的缩写,它是软件工程中的一种软件架构设计模式,它把软件系统分 为模型、视图和控制器三个基本部分

• View(视图) 指在应用程序中专门用来与浏览器进行交互,展示数据的资源.
• Model(模型) 是应用程序的主体部分,用来处理程序中数据逻辑的部分.
• Controller(控制器)可以理解为一个分发器,用来决定对于视图发来的请求,需要用哪一个模型
来处理,以及处理完后需要跳回到哪一个视图。即用来连接视图和模型
1.2 什么是Spring MVC?
总的来说:Spring MVC就是一个实现了MVC模式的Web框架
其实,在前面学习创建Spring Boot项目时,勾选的Spring Web框架其实就是Spring Web MVC框架:
可能有人会有疑问:为什么创建的是Spring Boot项目,又会创建了一个Spring MVC项目?其实,Spring Boot是实现Spring MVC的一种方式,Spring Boot可以添加多种依赖,通过不同的依赖实现不同的功能,Spring Boot正是通过添加Spring Web依赖来实现Web功能的。
Spring在实现MVC时,也做出了一些改变:

不过现在,我们在创建Spring MVC项目时已经不处理视图相关的问题了,这些问题一般交由前端人员进行处理。
二、练习Spring MVC
Spring MVC是一个Web框架,当用户在浏览器输入url后,Spring MVC可以感知到用户发送的请求并返回响应,因此,学习Spring MVC主要学习以下三个方面:
<1>建立连接:将用户(浏览器)和 Java 程序连接起来,也就是访问一个地址能够用到我们的 Spring 程序。
<2>请求 :用户请求的时候会带一些参数,在程序中要想办法获取到参数, 所以请求这块主要是获取参数的功能.
<3>响应: 执行了业务逻辑之后,要把程序执行的结果返回给用户, 也就是响应.
2.1 创建Spring MVC项目
创建Spring MVC项目的方法与前面创建Spring Boot显目的方法一样:只需勾选上Spring Web依赖即可
2.2 建立连接
在 Spring MVC 中使用 @RequestMapping 来实现 URL 路由映射 ,也就是浏览器连接程序的作用,在启动类(后缀名为Application)所在目录创建一个UserController类来实现用户通过浏览器和程序的互动:
2.2.1 @RequestMapping注解介绍
@RequestMapping 是 Spring Web MVC 应用程序中最常被用到的注解之一,它是用来注册接口的路由映射的.
表示服务收到请求时, 路径为 /sayHi 的请求就会调用 sayHi 这个方法的代码.
路由映射: 当用户访问⼀个 URL 时, 将用户的请求对应到程序中某个类的某个方法的过程就叫路由映射.
既然@RequesMapping这个注解已经可以达到访问的目的,那为什么还要@RestController注解呢?
如果我们去掉@RestController注解,在来访问/sayHi这个路径,会出现如下情况:
可以看到程序报错,找不到该页面,这就是@RestController注解的作用,Spring会对所有的类进行扫描, 如果类加了注解@RestController, Spring才会去看这个类里面的方法
有没有加 @RequestMapping 这个注解, 这个注解还有更多细节,后面会慢慢了解到,现在我们只需要知道,如果要通过浏览器访问程序的某个方法,那么这个方法所在的类一定要加上@Restcontroller注解
2.2.2 @RequestMapping的使用
@RequestMapping既可以修饰类,也可以修饰方法,当修饰类和方法时,访问的地址为类路径+方法路径。
@RequestMapping标识⼀个类:设置映射请求的请求路径的初始信息
@RequestMapping标识⼀个方法:设置映射请求请求路径的具体信息
!!!@RequestMapping中的URL路径可以是多重路径,如果是多重路径,路径中的 / 一定不能省略,如果不是则可以省略。
2.2.3 @RequestMapping是GET请求还是POST请求?
<1>GET请求:通过浏览器发送的请求都是GET请求,因此@RequestMapping支持GET请求
<2>POST请求:我们可以通过form表单来构造请求:
接下来通过浏览器访问test.html,并点击提交:
可以看到,@RequestMapping也支持post请求。
那么,如何指定请求类型?
<1>通过@RequestMapping的method属性设置;
<2>将@RequestMapping注解改为@GetMapping或@PostMapping或其它(以此类推)
如图:
我们发现,如果想要通过其它请求方式访问我们的代码,就需要通过编写前端代码来实现,非常麻烦,这里推荐使用Postman来发送请求,它可以指定任意请求方式,如有疑问,可参考如下文章:最简单最适合纯小白的postman使用方法(测试接口的不二利器)(从介绍到下载到使用的详细教程)_postman连接java后端-CSDN博客
2.3 请求
2.3.1 传递单个参数
接下来,使用Postman发送请求:
可以看到,getInt的类型为包装类,其实也可以使用普通类型来接收,但是如果此时传递来的参数是null,那么将会报错,如:
而包装类,则不会出现这种情况:
因此,使用基本类型来接收参数时,接收的参数一定不能为null,且使用基本类型接收参数,这个参数必传,否则也会报错,参数类型不匹配,也会报错(相当于接收参数为null)。
2.4.2 传递多个参数
传递多个参数与传递单个参数类似,直接使用多个形参接收即可:
需要注意的是,当有多个参数时,前后端进行参数匹配时,是以参数的名称进行匹配的,因此参数的位置是不影响后端获取参数的结果.
2.4.3 传递对象
<1> 先创建一个类
<2> 创建一个UserInfo对象,用于接收参数
<3>在Postman中传参,传参方式和传递多个参数一样
当需要传递的参数个数很多时,就可以改为使用对象来接收
2.4.4 后端参数重命名
某些特殊的情况下,前端传递的参数 key 和我们后端接收的 key 可以不⼀致,比如前端传递了⼀个 time 给后端,而后端是使用 createtime 字段来接收的,这样就会出现参数接收不到的情况,如果出现这种情况,我们就可以使⽤ @RequestParam 来重命名前后端的参数值 ,如:
在形参前面加上@RequestParam注解,并在注解中写入一个属性 “q“ ,表示前端传递过来的参数q的值,将会被keyWord接收:
同时,此时的 q 默认为一个必传参数,如果不传将会报错:
如果想要将 q 设置为非必传参数,可以将@RequestParama中的属性requird置为false即可
2.4.5 传递数组
Spring MVC可以自动绑定数组的赋值,故可以直接使用数组接收
2.4.6 传递集合
Spring MVC不可以自动绑定集合的赋值,如果需要传递集合,需要使用@RequestParama注解进行参数绑定。
接下来在Postman进行传参即可:
如果不加@RequestMapping注解,将会报错:
2.4.7 传递Json对象
传递Json对象需要使用@RequestBody注解,它能够将传递过来的Josn数据转换为相应的对象
接下来在Postman中传递Json数据:
如果不使用@RequestBody注解,赋值不会成功:
2.4.8 获取URL中参数@PathVariable
获取url上的参数需要使用@PathVariable注解,这个注解主要作用在请求url路径上的数据绑定。
在Postman中发送请求:
参数对应关系如下:
2.4.9 上传文件@RequestPart
使用Postman发送请求:
然后观察相应路径,看是否上传成功即可
2.4.10 获取Cookie/Session
一、Cookie
由于HTTP属于“无状态”协议(两次通信之间没有直接联系),但是在实际开发中,需要知道两次请求之间的关系(如登录某个网站后,下一次访问服务器能知道这次的请求是否已经登录过了)
图中的令牌通常就存储在Cookie中
此时在服务器这边就需要记录"令牌"信息, 以及令牌对应的用户信息, 这个就是 Session 机制所做的工作.
二、Session
Session表示一个会话(客户与服务器之间的不中断的请求响应),它是服务器用来区分不同的请求属于哪个会话的,是服务器为了保存用户信息而创建的⼀个特殊的对象。
Session的本质是一个“哈希表”,Key就是SessionId,value就是保存的用户信息
上图中对应的流程如下:
三、Session和Cookie的区别
<1>Cookie是客户端保存用户信息的一种机制,Session是服务器保存用户信息的一种机制
<2>Cookie和Session通过SessionId关联起来
<3>Cookie和Session通常在一起使用,但并不一定:
1. Cookie可以用来保存其它数据,不一定是用户身份信息和SessionId
2.SessionId也不一定通过Set-Cookie传递给Cookie,也可以通过URL传递
四、获取Cookie
<1>传统方式获取Cookie
如果想要获取HTTP请求中的任何数据,都可以使用HttpServletRequest这个类创建的对象来获取,如:
接下来在浏览器中访问:
由于现在Cooike中还没有值,因此控制台没有输出相关信息:
接下来,通过F12手动添加一个Cookie:
手动添加后,刷新页面查看idea控制台,此时Cookie的值已被打印
!!!可以看到,Cookie是可以伪造的,因此在后端使用Cookie时,需要进行校验
<2>通过注解@CookieValue获取Cookie
通过浏览器访问:
通过注解可以更简单地获取到Cookie的值,但是缺点是一次只能获取一个Cookie(一次请求携带的Cookie可能有多个)
五、Session的存储
Session时服务端的机制,需要先存储,才能获取
getSession方法带有一个布尔类型的参数,默认情况下为true,表示如果没有Session对象,就会创建一个空的Session对象,如果改为false,则没有Session的情况下会返回null,通过Session对象的setAttribute保存用户信息。
六、Session的获取
<1>通过HttpServletRequest获取
getSession是通过SessionId来获取Session的,因此需要先通过访问setSession方法存储Session,再访问getSession方法,其对应的具体流程为:
(1)首次创建会话时(即通过setSession的getSession方法获取到session对象),服务器会生成JSESSIONID并通过Set-Cookie发送给客户端;
(2)客户端在后续请求中通过Cookie字段发送JSESSIONID;
(3)服务器通过JSESSIONID找到对应的HttpSession对象,供getSession方法使用。
现在先通过浏览器访问setSession方法,通过F2观察:
可以看到,SessionId已经通过Set-Cookie保存到Cookie中了,接下来通过浏览器访问getSession方法:
成功获取到用户信息,如果不先调用setSession方法,返回用户未登录:
<2>通过HttpSession获取Session
<3>通过注解获取Session
2.4.11 获取Header
<1>通过HttpServletRequest获取
同理,也可以通过getHeader获取到Header中的Content-Length、Content-Type等属性。
通过Postman访问getHeader方法:
<2>通过注解获取Header
2.4 响应
在前面的请求代码中,也已经设置了响应数据, 响应结果可以是数据, 也可以是静态页面,也可
以针对响应设置状态码, Header信息等。
2.4.1 返回静态页面
(1)首先,在static文件夹中创建index.html文件
(2)编写后端代码
(3)在浏览器中访问
可以看到,成功返回了一个页面。
但是在这里有几个需要注意的问题:
<1>一般后端代码中返回的页面的路劲最前面需要加上“ / ”,前端一般不加
<2>如果后端返回的是一个页面,需要将注解@RestController改为@Controller,因为@RestController表示返回结果是数据,@Controller表示返回结果是页面,如果在上面的代码中还是使用@RestController注解的话,那么会将"/index.html"解析为数据,如:
<3>如果类注解使用了@Controller,而类中有一个方法需要返回数据,可以在方法前加上注解@ResponseBody,可以理解为@RestController包含了@Controller和@responseBody两个注解。
!!!到了现在Spring MVC已经不再返回视图,而是返回显示视图需要的数据,将返回视图的任务交由前端完成。
2.4.2 返回数据@ResponseBody
这里的类注解为Controller,故方法前需要加上注解@ResponseBody表示返回数据
如果不加@ResponseBody,会报404
!!!@ResponseBody 既是类注解, 又是方法注解 如果作用在类上, 表示该类的所有方法, 返回的都是数据, 如果作用在方法上, 表示该方法返回的是数据.
2.4.3 返回HTML代码片段
在返回的数据中,如果又HTML代码片段,会被浏览器解析
在浏览器访问:
通过Fiddler抓包观察,发现Content-Type的类型为text/html,表示响应Body中的数据格式为HTML
如果不想返回结果被解析为HTML,可以通过@RequestMapping的produces属性将Content-Type修改为text/plain,如:
通过浏览器访问:
再次通过Fillder抓包:
可以看到Content-Type已被修改
2.4.4 返回JSON
HTTP协议不支持直接传输Java对象或List等,如果后端返回了Java对象或List,就会转换为JSON,如:
通过Postman访问:
2.4.5 设置状态码
通过HttpServletResponse修改状态码:
通过浏览器访问:
使用Fiddler抓包并观察:
虽然请求成功的状态码一般默认为200,但是后端人员也可以将其设置为404,但是并不符合人HTTP规范
2.4.6 设置Header
<1>设置Content-type
前面已近提到,可以通过@RequestMapping注解的produces属性进行修改
<2>自定义Header片段
使用HttpServletResponse对象进行修改
通过浏览器访问并使用Fiddler抓包观察:
成功在Header中添加新的字段
相关文章:
Spring Web MVC项目的创建及使用
一、什么是Spring Web MVC? Spring Web MVC 是基于 Servlet API 构建的原始 Web 框架,从⼀开始就包含在 Spring 框架中,通常被称为Spring MVC。 1.1 MVC的定义 MVC 是 Model View Controller 的缩写,它是软件工程中的一种软件架构…...
32.日常算法
1.最大子数组和 题目来源 给你一个整数数组 nums ,请你找出一个具有最大和的连续子数组(子数组最少包含一个元素),返回其最大和。子数组是数组中的一个连续部分。 示例 1: 输入:nums [-2,1,-3,4,-1,2,1,…...
MySQL的底层原理与架构
前言 了解MySQL的架构和原理对于很多的后续很多的操作会有很大的帮助与理解。并且很多知识都与底层架构相关联。 了解MySQL架构 通过上面的架构图可以得知,Server层中主要由 连接器、查询缓存、解析器/分析器、优化器、执行器 几部分组成的,下面将主要…...
python Excel 表读取合并单元格以及清除空格符
读取合并单元格并保留合并信息 读取合并单元格并保留合并信息清除各单元格的空格和换行符,并去除列名中的空格和换行符 读取合并单元格并保留合并信息 当我们只是使用 pandas 的 read_excel 方法读取 Excel 文件时,我们可能会遇到一个很棘手的问题&…...
Node.js 实现简单爬虫
介绍 爬虫是一种按照一定的规则,自动地抓取万维网信息的程序或者脚本。 本文将使用 Nodejs 编写一个简单的爬虫脚本,爬取一个美食网站,获取菜品的标题和图片链接,并以表格的形式输出。 准备工作 1、初始化项目 首先࿰…...
【Pytorch】nn.RNN、nn.LSTM 和 nn.GRU的输入和输出形状
nn.RNN、nn.LSTM 和 nn.GRU的输入和输出形状 输入形状通用输入参数特殊情况(LSTM) 输出形状nn.RNN 和 nn.GRUnn.LSTM 代码示例 输入形状 通用输入参数 这三个模块通常接收以下两种形式的输入: 输入序列:形状为 (seq_len, batch…...
代码随想录算法训练营第三十一天| 回溯算法04
491. 递增子序列 题目: 代码随想录 视频讲解:回溯算法精讲,树层去重与树枝去重 | LeetCode:491.递增子序列_哔哩哔哩_bilibili 这题需要注意的点: 1. path长度在2以上才放入最终结果 2. 需要记录已经使用过的数字&am…...
2024.1版android studio创建Java语言项目+上传gitee
1.在gitee上创建仓库 Gitee 创建仓库并邀请成员指南_gitee创建仓库邀请成员-CSDN博客 见1 2.新建android studio项目 3.在Android studio配置gitee Android Studio提交代码到gitee仓库_android log in to gitee-CSDN博客 其中的一二步 p.s.添加gitee账户选择password时&a…...
React 打印插件 -- react-to-print
一、安装依赖 npm install react-to-print 二、使用 import { useReactToPrint } from "react-to-print"; import React, { useRef, forwardRef } from react;const Content () > {const contentRef useRef(null);const reactToPrintFn useReactToPrint({ c…...
opentelemetry-collector 配置elasticsearch
一、修改otelcol-config.yaml receivers:otlp:protocols:grpc:endpoint: 0.0.0.0:4317http:endpoint: 0.0.0.0:4318 exporters:debug:verbosity: detailedotlp/jaeger: # Jaeger supports OTLP directlyendpoint: 192.168.31.161:4317tls:insecure: trueotlphttp/prometheus: …...
SQL Server 数据库迁移到 MySQL 的完整指南
文章目录 引言一、迁移前的准备工作1.1 确定迁移范围1.2 评估兼容性1.3 备份数据 二、迁移工具的选择2.1 使用 MySQL Workbench2.2 使用第三方工具2.3 手动迁移 三、迁移步骤3.1 导出 SQL Server 数据库结构3.2 转换数据类型和语法3.3 导入 MySQL 数据库3.4 迁移数据3.5 迁移存…...
C# SQlite使用流程
前言 不是 MySQL 用不起,而是 SQLite 更有性价比,绝大多数的应用 SQLite 都可以满足。 SQLite 是一个用 C 语言编写的开源、轻量级、快速、独立且高可靠性的 SQL 数据库引擎,它提供了功能齐全的数据库解决方案。SQLite 几乎可以在所有的手机…...
MySQL数据库 (三)- 函数/约束/多表查询/事务
目录 一 函数 (一 字符串函数 (二 数值函数 (三 日期函数 (四 流程函数 二 约束 (一 概述 (二 约束演示 (三 外键约束 三 多表查询 (一 多表关系 1 一对多(多对一) 2 多对多 3 一对一 (二 多表查询概述 (三 内连接 1 查询语法 2 代码实…...
【玩转 Postman 接口测试与开发2_018】第14章:利用 Postman 初探 API 安全测试
《API Testing and Development with Postman》最新第二版封面 文章目录 第十四章 API 安全测试1 OWASP API 安全清单1.1 相关背景1.2 OWASP API 安全清单1.3 认证与授权1.4 破防的对象级授权(Broken object-level authorization)1.5 破防的属性级授权&a…...
攻防世界baigeiRSA
打开题目附件 import libnum from Crypto.Util import number from secret import flagsize 128 e 65537 p number.getPrime(size) q number.getPrime(size) n p*qm libnum.s2n(flag) c pow(m, e, n)print(n %d % n) print(c %d % c)n 8850300144784503160345704866…...
12.7 LangChain代理系统Agents深度解析:构建自主决策的智能体应用
LangChain代理系统Agents深度解析:构建自主决策的智能体应用 一、代理系统的核心价值 代理系统是大模型应用的决策中枢,通过动态工具调度和任务分解,突破传统链式流程的三大局限: 动态规划:根据实时反馈调整执行路径工具集成:无缝对接500+外部系统API认知迭代:通过记忆…...
解释一下数据库中的事务隔离级别,在 Java 中如何通过 JDBC设置事务隔离级别?
数据库中的事务隔离级别是用于控制并发事务之间相互影响的一种机制。 它定义了事务之间的可见性和影响范围,常见的隔离级别包括: 读未提交(Read Uncommitted):最低的隔离级别,事务中的修改即使没有提交也…...
[NKU]C++安装环境 VScode
bilibili安装教程 vscode 关于C/C的环境配置全站最简单易懂!!大学生及初学初学C/C进!!!_哔哩哔哩_bilibili 1安装vscode和插件 汉化插件 2安装插件 2.1 C/C 2.2 C/C Compile run 2.3 better C Syntax 查看已…...
Node.js 环境配置
什么是 Node.js Node.js 是一个基于 Chrome V8 JavaScript 引擎的 JavaScript 运行时环境,它允许你在服务器端运行 JavaScript。传统上,JavaScript 主要用于浏览器中的前端开发,而 Node.js 使得 JavaScript 也能够在服务器上执行,…...
1Panel应用推荐:WordPress开源博客软件和内容管理系统
1Panel(github.com/1Panel-dev/1Panel)是一款现代化、开源的Linux服务器运维管理面板,它致力于通过开源的方式,帮助用户简化建站与运维管理流程。为了方便广大用户快捷安装部署相关软件应用,1Panel特别开通应用商店&am…...
C++:string类的模拟实现
目录 1.引言 2.C模拟实现 2.1模拟实现构造函数 1)直接构造 2)拷贝构造 2.2模拟实现析构函数 2.3模拟实现其他常规函数 1)c_str函数 2)size函数 3)begin/end函数 4)reserve函数 5)re…...
DMZ区的作用和原则
DMZ(Demilitarized Zone,非军事化区)是网络安全架构中一个重要的概念,其主要作用和原则如下: DMZ的作用 隔离风险 DMZ作为内外网络之间的缓冲区,能够有效隔离外部网络的攻击风险。将对外提供服务的服务器&…...
21.命令模式(Command Pattern)
定义 命令模式(Command Pattern) 是一种行为型设计模式,它将请求封装成一个对象,从而使您可以使用不同的请求、队列、日志请求以及支持撤销操作等功能。命令模式通过将请求(命令)封装成对象,使…...
如何将本地 Node.js 服务部署到宝塔面板:完整的部署指南
文章简介: 将本地开发的 Node.js 项目部署到线上服务器是开发者常见的工作流程之一。在这篇文章中,我将详细介绍如何将本地的 Node.js 服务通过宝塔面板(BT 面板)上线。宝塔面板是一个强大的服务器管理工具,具有简洁的…...
4.3 线性回归的改进-岭回归/4.4分类算法-逻辑回归与二分类/ 4.5 模型保存和加载
4.3.1 带有L2正则化的线性回归-岭回归 岭回归,其实也是一种线性回归,只不过在算法建立回归方程的时候1,加上正则化的限制,从而达到解决过拟合的效果 4.3.1.1 API 4.3.1.2 观察正则化程度的变化,对结果的影响 正则化力…...
Mac 部署Ollama + OpenWebUI完全指南
文章目录 💻 环境说明🛠️ Ollama安装配置1. 安装[Ollama](https://github.com/ollama/ollama)2. 启动Ollama3. 模型存储位置4. 配置 Ollama 🌐 OpenWebUI部署1. 安装Docker2. 部署[OpenWebUI](https://www.openwebui.com/)(可视化…...
工业物联网平台-视频识别视频报警新功能正式上线
前言 视频监控作为中服云工业物联网平台4.0的功能已经上线运行。已为客户服务2年有余,为客户提供多路视频、实时在线监视和控制能力。服务客户实时发现现场、产线、设备出现随机故障、事故等,及时到场处理维修。 视频识别&视频报警新功能当前正式上…...
面试题 17.19. 消失的两个数字
文章目录 1.题目2.思路3.代码 1.题目 面试题 17.19. 消失的两个数字 给定一个数组,包含从 1 到 N 所有的整数,但其中缺了两个数字。你能在 O(N) 时间内只用 O(1) 的空间找到它们吗? 以任意顺序返回这两个数字均可。 示例 1: **输入:** […...
Licode简介及与SRS对比
Licode 是一个开源的 WebRTC 通信框架,专注于多人实时音视频互动(如视频会议),而 SRS 是一个通用的 流媒体服务器,支持直播、低延迟流分发等场景。以下是两者的详细对比和 Licode 的核心解析: 一、Licode 核心解析 1. 定位与设计目标 核心功能:基于 WebRTC 的多人实时音…...
mysql的cpu使用率100%问题排查
背景 线上mysql服务器经常性出现cpu使用率100%的告警, 因此整理一下排查该问题的常规流程。 1. 确认CPU占用来源 检查系统进程 使用 top 或 htop 命令,确认是否是 mysqld 进程导致CPU满载:top -c -p $(pgrep mysqld)2. 实时分析MySQL活动 …...

























接下来在Postman中传递Json数据:















































