JavaEE进阶学习:Spring MVC 程序开发

1.什么是 Spring MVC

Spring Web MVC 是基于Servlet API 构建的原始 Web 框架,从一开始就包含在Spring 框架中。它的正式名称 “Spring Web MVC” 来自其源模块的名称(Spring-webmvc),但它通常被称为“Spring MVC”。

从上述定义我们可以得出两个关键信息:

  1. Spring MVC 是⼀个 Web 框架。

  2. Spring MVC 是基于 Servlet API 构建的。

1.什么是 MVC

MVC 是 Model View Controller 的缩写,它是软件工程中的一种软件架构模式,它把软件系统分为模型、视图和控制器三个基本部分。

在这里插入图片描述

执行流程:

  1. 用户的请求首先先到 Controller
  2. Controller 将请求转发给 Model
  3. Model 处理业并将数据及结果给 Controller
  4. Controller 会将数据给 View 引擎
  5. View 转换数据生成最终的页面给用户
  • Model(模型)是应用程序中用处理应用程序数据逻辑的部分。通常模型对象负责在数据库中存取数据。
  • View(视图)是应用程序中处理数据显示的部分。通常视图是依据模型数据创建的。
  • Controller(控制器)是应⽤程序中处理用户交互的部分。通常控制器负责从视图读取数据,控制用户输入,并向模型发送数据。

2.MVC 和 Spring MVC 的关系

MVC 是一种思想,而 Spring MVC 是对 MVC 思想的具体实现。

总结来说,Spring MVC 是⼀个实现了 MVC 模式,并继承了 Servlet API 的 Web 框架。既然是 Web框架,那么当用户在浏览器中输入了 url 之后,我们的 Spring MVC 项⽬就可以感知到用户的请求。

2.Spring MVC 创建和连接

学习 Spring MVC 我们只需要掌握以下 3 个功能:

  1. 连接的功能:将用户(浏览器)和 Java 程序连接起来,也就是访问一个地址能够调用到我们的Spring 程序。
  2. 获取参数的功能:用户访问的时候会带一些参数,在程序中要想办法获取到参数。
  3. 输出数据的功能:执行了业务逻辑之后,要把程序执行的结果返回给用户。

对于 Spring MVC 来说,掌握了以上 3 个功能就相当于掌握了 Spring MVC。

Spring MVC 项目创建和 Spring Boot 创建项目相同(Spring MVC 使用 Spring Boot 的方式创建),在创建的时候选择 Spring Web 就相当于创建了 Spring MVC 的项目。

在 Spring MVC 中使用 @RequestMapping 来实现 URL 路由映射,也就是浏览器连接程序的作用。

接下来要实现的功能是访问地址:http://localhost:8080/user/hi,能打印“hello,spring mvc”信息。

1.实现客户端和程序之间的连接

1.@RequestMapping

package com.example.project.controller;import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;/*** @projectName: Project* @package: com.example.project.controller* @className: TestController* @author: 王嘉辉* @description:* @date: 2023/12/23 18:15* @version: 1.0*/
@RestController
@RequestMapping("/test")
public class TestController {@RequestMapping("/sayhi")public String sayHi() {return "Hello Spring MVC";}
}

在这里插入图片描述

在这里插入图片描述

@RequestMapping 还可以指定 GET/POST 方法类型

在这里插入图片描述

在这里插入图片描述

2.@GetMapping

实现 HTTP 链接,但只支持 GET 类型的请求

package com.example.project.controller;import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RestController;/*** @projectName: Project* @package: com.example.project.controller* @className: TestController* @author: 王嘉辉* @description:* @date: 2023/12/23 18:15* @version: 1.0*/
@RestController
@RequestMapping("/test")
public class TestController {@RequestMapping(path = "/sayhi",method = RequestMethod.GET)public String sayHi() {return "Hello Spring MVC !!!";}@GetMapping("/sayhi2")public String sayHi2() {return "Hello Spring MVC2 !!!";}
}

在这里插入图片描述

get 请求的 3 种写法:

// 写法1
@RequestMapping("/index")
// 写法2
@RequestMapping(value = "/index",method = RequestMethod.GET)
// 写法3
@GetMapping("/index")

3.@PostMapping

实现 HTTP 链接,但只支持 POST 类型的请求

package com.example.project.controller;import org.springframework.web.bind.annotation.*;/*** @projectName: Project* @package: com.example.project.controller* @className: TestController* @author: 王嘉辉* @description:* @date: 2023/12/23 18:15* @version: 1.0*/
@RestController
@RequestMapping("/test")
public class TestController {@RequestMapping(path = "/sayhi",method = RequestMethod.GET)public String sayHi() {return "Hello Spring MVC !!!";}@GetMapping("/sayhi2")public String sayHi2() {return "Hello Spring MVC2 !!!";}@PostMapping("/sayhi3")public String sayHi3() {return "Hello Spring MVC3 !!!";}
}

在这里插入图片描述

post 请求的 3 种写法:

// 写法1
@RequestMapping("/index")
// 写法2
@RequestMapping(value = "/index",method = RequestMethod.POST)
// 写法3
@PostMapping("/index")

2.获取参数

1.传递单个参数

package com.example.project.controller;import jakarta.servlet.http.HttpServletRequest;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;/*** @projectName: Project* @package: com.example.project.controller* @className: TestController2* @author: 王嘉辉* @description:* @date: 2023/12/23 18:44* @version: 1.0*/
@RestController
@RequestMapping("/test2")
public class TestController2 {@RequestMapping("/getname")public String getName(HttpServletRequest request) {return "Name: " + request.getParameter("name");}
}

在这里插入图片描述

package com.example.project.controller;import jakarta.servlet.http.HttpServletRequest;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;/*** @projectName: Project* @package: com.example.project.controller* @className: TestController2* @author: 王嘉辉* @description:* @date: 2023/12/23 18:44* @version: 1.0*/
@RestController
@RequestMapping("/test2")
public class TestController2 {@RequestMapping("/getname")public String getName(HttpServletRequest request) {return "Name: " + request.getParameter("name");}@RequestMapping("/getname2")public String getName2(String name) {return "Name: " + name;}
}

在这里插入图片描述

2.传递多个参数

package com.example.project.controller;import jakarta.servlet.http.HttpServletRequest;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;/*** @projectName: Project* @package: com.example.project.controller* @className: TestController2* @author: 王嘉辉* @description:* @date: 2023/12/23 18:44* @version: 1.0*/
@RestController
@RequestMapping("/test2")
public class TestController2 {@RequestMapping("/getname")public String getName(HttpServletRequest request) {return "Name: " + request.getParameter("name");}@RequestMapping("/getname2")public String getName2(String name) {return "Name: " + name;}@RequestMapping("/getname3")public String getName3(String name,Integer age) {return "Name: " + name + " age: " + age;}
}

在这里插入图片描述

3.传递对象

package com.example.project.controller;import com.example.project.model.User;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;/*** @projectName: Project* @package: com.example.project.controller* @className: UserController* @author: 王嘉辉* @description:* @date: 2023/12/23 18:59* @version: 1.0*/
@RestController
@RequestMapping("/user")
public class UserController {@RequestMapping("/add")public User add(User user) {return user;}
}
package com.example.project.model;import lombok.Data;/*** @projectName: Project* @package: com.example.project.model* @className: User* @author: 王嘉辉* @description:* @date: 2023/12/23 18:58* @version: 1.0*/
@Data
public class User {private int id;private String name;private String password;private int age;
}

在这里插入图片描述

4.后端参数重命名

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

如果我们的实际业务前端的参数是⼀个非必传的参数,我们可以通过设置 @RequestParam 中的 required=false 来避免不传递时报错

5.获取前端的 JSON 对象

在这里插入图片描述

在这里插入图片描述

6.获取URL中参数

@RequestMapping("/detail/{aid}")public Integer detail(@PathVariable("aid") Integer aid) {return aid;}

在这里插入图片描述

7.上传文件

	@RequestMapping("/upload")public String upload(@RequestPart("myfile")MultipartFile file) throws IOException {String path = "E:\\study\\Gitee\\test_c\\test-2023-12-23\\Project\\file\\img.jpg";//保存文件file.transferTo(new File(path));return path;}

在这里插入图片描述
在这里插入图片描述

在这里插入图片描述

	@RequestMapping("/upload")public String upload(@RequestPart("myfile")MultipartFile file) throws IOException {//1.生成一个唯一的id | UUID = 全球唯一ID -> MAC + 随机种子 + 加密算法String name = UUID.randomUUID().toString().replace("-","");//2.得到源文件后缀名name += file.getOriginalFilename().substring(file.getOriginalFilename().lastIndexOf("."));String path = "E:\\study\\Gitee\\test_c\\test-2023-12-23\\Project\\file\\" + name;//保存文件file.transferTo(new File(path));return path;}

保证不被覆盖

在这里插入图片描述

8.获取Cookie/Session/header

获取 Request 和 Response 对象

@RequestMapping("/param10")
public String param10(HttpServletResponse response, HttpServletRequest requ
est) {String name = request.getParameter("name");// 获取所有 cookie 信息Cookie[] cookies = request.getCookies();return name + " 你好.";
}

传统获取 header/cookie

@RequestMapping("/param10")
@ResponseBody
public String param10(HttpServletResponse response, HttpServletRequest requ
est) {String name = request.getParameter("name");// 获取所有 cookie 信息Cookie[] cookies = request.getCookies();String userAgent = request.getHeader("User-Agent");return name + ":"+userAgent;
}

简洁的获取 Cookie—@CookieValue

	@RequestMapping("/cookie")public String cookie(@CookieValue(value = "java",required = false) String ck) {return "cookie:" + ck;}

简洁获取 Header—@RequestHeader

@RequestMapping("/header")
@ResponseBody
public String header(@RequestHeader("User-Agent") String userAgent) {return "userAgent:" + userAgent;
}

session 存储和获取

package com.example.project.controller;import com.example.project.model.User;
import jakarta.annotation.PostConstruct;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpSession;
import org.springframework.web.bind.annotation.*;
import org.springframework.web.multipart.MultipartFile;import java.io.File;
import java.io.IOException;
import java.util.UUID;/*** @projectName: Project* @package: com.example.project.controller* @className: UserController* @author: 王嘉辉* @description:* @date: 2023/12/23 18:59* @version: 1.0*/
@RestController
@RequestMapping("/user")
public class UserController {private static String _SESSION_KEY = "SESSION_KEY";@RequestMapping("/add")public User add(User user) {return user;}@RequestMapping("/name")public String name(@RequestParam(value = "n",required = false)String name) {return name;}@RequestMapping("/add_json")public User addByJson(@RequestBody User user) {return user;}@RequestMapping("/detail/{aid}")public Integer detail(@PathVariable("aid") Integer aid) {return aid;}@RequestMapping("/upload")public String upload(@RequestPart("myfile")MultipartFile file) throws IOException {//1.生成一个唯一的id | UUID = 全球唯一ID -> MAC + 随机种子 + 加密算法String name = UUID.randomUUID().toString().replace("-","");//2.得到源文件后缀名name += file.getOriginalFilename().substring(file.getOriginalFilename().lastIndexOf("."));String path = "E:\\study\\Gitee\\test_c\\test-2023-12-23\\Project\\file\\" + name;//保存文件file.transferTo(new File(path));return path;}@RequestMapping("/cookie")public String Cookie(@CookieValue(value = "java",required = false) String ck) {return "cookie:" + ck;}@RequestMapping("/set_sess")public String setSess(HttpServletRequest request) {HttpSession session = request.getSession();if(session != null) {session.setAttribute(_SESSION_KEY,"张三");return "session set success";}else {return "session set fail";}}@RequestMapping("/get_sess")public String getSess(@SessionAttribute(required = false,value = "SESSION_KEY") String name) {return name;}
}

在这里插入图片描述

在这里插入图片描述

3.返回参数

通过上面的学习我们知道,默认请求下无论是 Spring MVC 或者是 Spring Boot 返回的是视图(xxx.html),而现在都是前后端分离的,后端只需要返给给前端数据即可,这个时候我们就需要使用@ResponseBody 注解了。

package com.example.project.controller;import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;/*** @projectName: Project* @package: com.example.project.controller* @className: TestController3* @author: 王嘉辉* @description:* @date: 2023/12/23 22:04* @version: 1.0*/
@Controllerpublic class TestController3 {@RequestMapping("/index")public String index() {return "hello.html";}
}
<!doctype html>
<html lang="en">
<head><meta charset="UTF-8"><meta name="viewport"content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0"><meta http-equiv="X-UA-Compatible" content="ie=edge"><title>Document</title>
</head>
<body><h1>hello,i am hello</h1>
</body>
</html>

在这里插入图片描述

package com.example.project.controller;import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;/*** @projectName: Project* @package: com.example.project.controller* @className: TestController3* @author: 王嘉辉* @description:* @date: 2023/12/23 22:04* @version: 1.0*/
@Controller
@ResponseBody
public class TestController3 {@RequestMapping("/index")public String index() {return "hello.html";}
}

在这里插入图片描述

4.请求转发或请求重定向

return 不但可以返回一个视图,还可以实现跳转,跳转的方式有两种:

forward: 是请求转发;

redirect:请求重定向。

// 请求重定向
@RequestMapping("/index")
public String index(){return "redirect:/index.html";
}
// 请求转发
@RequestMapping("/index2")
public String index2(){return "forward:/index.html";
}

forward(请求转发)和 redirect(请求重定向)的区别,举例来说,例如,你告诉你妈妈,你想吃辣条,如果你妈妈,说好,我帮你去买,这就是 forward 请求转发;如果你妈妈让你自己去买,那么就是请求 redirect 重定向。

forward 和 redirect 具体区别如下:

  1. 请求重定向(redirect)将请求重新定位到资源;请求转发(forward)服务器端转发。
  2. 请求重定向地址发生变化,请求转发地址不发生变化。
  3. 请求重定向与直接访问新地址效果⼀直,不存在原来的外部资源不能访问;请求转发服务器端转发有可能造成原外部资源不能访问。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/241209.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

如何改善与 Next Paint (INP) 的交互

但谷歌也会关注访问者到达后你的网站体验有多好。 在过去的几年里&#xff0c;谷歌已经彻底改变了哪些页面体验信号被收集并用作排名因素。 在引入核心网络指标后&#xff0c;谷歌逐渐调整了它们的衡量方式&#xff0c;以便更好地反映真实的用户体验。 然而&#xff0c;随着…

Unity中获取时间戳、日期、时间、毫秒、秒以相互转换、自定义格式时间

Unity中获取时间戳、日期、时间、毫秒、秒以相互转换、自定义格式时间 介绍时间戳是什么什么时候用时间戳 获取时间获取当前时间获取时间戳日期转时间戳时间戳转日期将时间戳转换为多久之前星期自定义格式时间 总结 介绍 这里附带一个时间戳和时间转换的网址 时间戳是什么 时…

【Spring实战】01 配置单数据源

文章目录 1. 定义2. 准备3. 打印连接信息4. 实战1&#xff09;创建表2&#xff09;添加数据3&#xff09;查询数据3&#xff09;执行 5. 详细代码总结 在我们常见的应用程序中&#xff0c;与数据库的交互是不可避免的一部分。Spring 提供了简单而强大的数据访问抽象&#xff0c…

九州金榜|家庭教育幼小衔接家长如何做?

孩子从幼儿园升入小学&#xff0c;很多家长会非常忧虑&#xff0c;进入小学便是孩子学校生涯正式开始&#xff0c;这个阶段作为家长会非常焦虑&#xff0c;会考虑孩子能不能适应小学生活&#xff1f;学习跟不跟得上&#xff0c;一般这个时候&#xff0c;大部分家长就会考虑给孩…

数值分析期末复习

第一章 科学计算 误差 解题步骤 先求绝对误差: ∣ x − x ∗ ∣ |x - x^*| ∣x−x∗∣求相对误差限: ∣ x − x ∗ ∣ x ∗ \frac{|x\,\,-\,\,x^*|}{x^*} x∗∣x−x∗∣​求有效数字 ∣ x − x ∗ ∣ 需要小于它自身的半个单位 |x-x^*|\text{需要小于它自身的半个单位} ∣…

Python入门学习篇(六)——for循环while循环

1 for循环 1.1 常规for循环 1.1.1 语法结构 for 变量名 in 可迭代对象:# 遍历对象时执行的代码 else:# 当for循环全部正常运行完(没有报错和执行break)后执行的代码1.1.2 示例代码 print("----->学生检查系统<------") student_lists["张三",&qu…

MLX vs MPS vs CUDA:苹果新机器学习框架的基准测试

如果你是一个Mac用户和一个深度学习爱好者&#xff0c;你可能希望在某些时候Mac可以处理一些重型模型。苹果刚刚发布了MLX&#xff0c;一个在苹果芯片上高效运行机器学习模型的框架。 最近在PyTorch 1.12中引入MPS后端已经是一个大胆的步骤&#xff0c;但随着MLX的宣布&#x…

在Excel中,如何简单快速地删除重复项,这里提供详细步骤

当你在Microsoft Excel中使用电子表格时&#xff0c;意外地复制了行&#xff0c;或者如果你正在制作其他几个电子表格的合成电子表格&#xff0c;你将遇到需要删除的重复行。这可能是一项非常无脑、重复、耗时的任务&#xff0c;但有几个技巧可以让它变得更简单。 删除重复项 …

Android Canvas画布saveLayer与对应restoreToCount,Kotlin

Android Canvas画布saveLayer与对应restoreToCount&#xff0c;Kotlin private fun mydraw() {val originBmp BitmapFactory.decodeResource(resources, R.mipmap.pic).copy(Bitmap.Config.ARGB_8888, true)val newBmp Bitmap.createBitmap(originBmp.width, originBmp.heigh…

【Win10安装Qt6.3】安装教程_保姆级

前言 Windows系统安装Qt4及Qt5.12之前版本和安装Qt.12之后及Qt6方法是不同的 &#xff1b;因为之前的版本提供的有安装包&#xff0c;直接一路点击Next就Ok了。但Qt5.12版本之后&#xff0c;Qt公司就不再提供安装包了&#xff0c;不论是社区版&#xff0c;专业版等&#xff0c…

并发控制工具类CountDownLatch、CyclicBarrier、Semaphore

并发控制工具类CountDownLatch、CyclicBarrier、Semaphore 1.CountDownLatch 可以使一个或多个线程等待其他线程各自执行完毕后再执行。 CountDownLatch 是多线程控制的一种工具&#xff0c;它被称为 门阀、 计数器或者闭锁。这个工具经常用来用来协调多个线程之间的同步&…

项目从vue2 升级vue3,项目大迁移 ,UI组件库更换

目录 背景描述 开发准备 第一步&#xff1a;升级环境 第二步&#xff1a;划分功能迁移顺序 第三步&#xff1a;详细了解需要迁移的业务页面 第四步&#xff1a;项目的一些配置的准备 详细开发流程 总结/分析&#xff1a; 背景描述 之前的版本&#xff1a;vue 2.6.8 i…

【PHY6222】绑定详解

1.函数详解 bStatus_t GAPBondMgr_SetParameter( uint16 param, uint8 len, void* pValue ) 设置绑定参数。 bStatus_t GAPBondMgr_GetParameter( uint16 param, void* pValue ) 获取绑定参数。 param&#xff1a; GAPBOND_PAIRING_MODE&#xff0c;配对模式&#xff0c;…

【postgres】8、Range 类型

文章目录 8.17 Range 类型8.17.1 内置类型8.17.2 示例8.17.3 开闭区间8.17.4 无穷区间 https://www.postgresql.org/docs/current/rangetypes.html 8.17 Range 类型 Range 类型&#xff0c;可以描述一个数据区间&#xff0c;有明确的子类型&#xff0c;而且子类型应该能被排序…

计算机网络——数据链路层(三)

前言: 前面我们已经对计算机网络的物理层有了一个大概的了解&#xff0c;今天我们学习的是物理层服务的上一层数据链路层&#xff0c;位于物理层和网络层之间。数据链路层在物理层提供的服务的基础上向网络层提供服务&#xff0c;其最基本的服务是将源自物理层来的数据可靠地传…

Mac使用Vmware Fusion虚拟机配置静态ip地址

一、设置虚拟机的网络为NAT 二、修改虚拟机的网络适配器网络 1、查看虚拟机的网卡 cd /etc/sysconfig/network-scripts#有些系统显示的是ens33&#xff0c;ens160等等 #不同的系统和版本&#xff0c;都会有所不同 #Centos8中默认是ens160,在RedHat/Centos7则为ens33 2、查看网…

Java语法---使用sort进行排序

目录 一、升序 二、降序 &#xff08;1&#xff09;类实现接口 &#xff08;2&#xff09;匿名内部类 三、自定义排序规则 四、集合中的sort排序 &#xff08;1&#xff09;升序 &#xff08;2&#xff09;降序 &#xff08;3&#xff09;自定义排序 一、升序 升序排…

C++内存管理和模板初阶

C/C内存分布 请看代码&#xff1a; int globalVar 1; static int staticGlobalVar 1; void Test() {static int staticVar 1;int localVar 1;int num1[10] { 1, 2, 3, 4 };char char2[] "abcd";const char* pChar3 "abcd";int* ptr1 (int*)mallo…

OpenHarmony之内核层解析~

OpenHarmony简介 技术架构 OpenHarmony整体遵从分层设计&#xff0c;从下向上依次为&#xff1a;内核层、系统服务层、框架层和应用层。系统功能按照“系统 > 子系统 > 组件”逐级展开&#xff0c;在多设备部署场景下&#xff0c;支持根据实际需求裁剪某些非必要的组件…

一周工作问题总结(2023.12.18-2023.12.22)

一周工作问题总结 1. 接口调用频率2. 汉字在数据库中占用字节问题3. Map在循环中修改自己的key与value4. Group BY5.递归同步数据6.代码移动Idea飘红 1. 接口调用频率 供应商给的接口可以每秒调用5-10次&#xff0c;那么我为了保险每秒调用5次&#xff0c;为了防止过度调用接口…