您现在的位置是:首页 >学无止境 >【Spring MVC】Web程序开发网站首页学无止境
【Spring MVC】Web程序开发
1. 什么是Spring MVC?
Spring MVC是Spring框架的一个Web模块,是一种基于MVC模型的Web应用开发框架
那MVC是啥呢?
MVC是一种软件架构思想,它将软件系统分为模型,视图,控制器三个基本部分:
- 模型Model:处理程序数据逻辑的部分,通常负责在数据库中存取数据
- 视图View:程序中处理数据显示部分
- 控制器Controller:程序中处理用户交互的部分
MVC是思想,Spring MVC是对MVC思想的具体实现,且继承了Servlet API的Web框架,故在浏览器中输入url后,Spring MVC可以感知到用户的请求
2. 如何创建Spring MVC项目?
在之前创建Spring Boot项目的基础上,添加Spring Web
依赖后,项目就变为Spring MVC项目了
学习Spring MVC的的以下三个功能:
- 连接的功能:将用户浏览器和Java程序绑定起来
- 获取参数的功能:获取用户访问时携带的参数
- 输出数据的功能:执行逻辑后,将程序执行结果返回给用户
3. 简单示例演示
//@Controller
//@ResponseBody //表示返回的是数据,不是页面
@RestController //@Controller + @ResponseBody
public class UserController {
@RequestMapping("/hello") //注册接口映射
public String printHello(){
return "hello Spring MVC";
}
}
类必须随着spring框架的启动而加载,所以加上@Controller注解交给容器管理
4. 路由接口映射
4.1 @RequestMapping
@RequestMapping 注解是用来注册路由接口映射的,也就是当用户输入url后,该请求能对应到程序中某个类的某个方法上
@RequestMapping可以修饰类也可以修饰方法
,当修饰类+方法时,路由地址为类+方法的组合地址,直接修饰方法时,路由地址就是方法上的地址,上述的示例就是直接修饰方法的
上述例子是浏览器输入url发送请求的,该请求是get请求,那使用@RequestMapping注解可以接收到post请求吗?
我们使用postman模拟发送post请求
发现@RequestMapping也可以接收到post请求,所以@RequestMapping既可以接收到get请求,又可以接收到post请求
那如何才能让方法只接收到特定的请求(只接收post或者只接收get)呢?
我们只需要在@RequestMapping上添加另一个参数,请求类型
@RequestMapping(value = "/hello",method = RequestMethod.POST) //注册接口映射
此时启动程序,再通过浏览器url访问
发现此时访问不了,那再通过postman模拟post请求
发现post请求依旧可以访问成功
4.2 @GetMapping和@PostMapping
- 方法上使用
@GetMapping
来进行路由映射,只能接收到get请求
- 方法上使用
@PostMapping
来进行路由映射,只能接收到post请求
@GetMapping("/hello")
public String printHello(){
return "hello Spring MVC";
}
@PostMapping("/hello")
public String printHello(){
return "hello Spring MVC";
}
@GetMapping和@PostMapping只能加在方法上,不能加在类
上
5. 获取参数
5.1 获取单个参数
@RequestMapping("/test")
public String print(String name){
return "hello "+name;
}
注意:传递参数的名称必须和方法里形参的名称对应起来
那不对应呢?发现获取不到参数
注意: 方法的
形参类型
尽量不要使用基础数据类型(int...)
,因为基础类型作为局部变量,当接收不到参数时,没有默认值,此时程序就会报错,使用包装类和引用类型
,就算接收不到参数,它们都有对应的默认值,程序也不会报错
5.2 获取多个参数
@RequestMapping("/login")
//参数的顺序无关紧要,只要名称对的上即可
public String login(String username,String password){
return "login success "+username+" "+password;
}
5.3 获取对象
@Data
public class User {
private String username;
private String password;
}
@RequestMapping("/reg")
public Object reg(User user){
return user;
}
注意:对象的属性名称也要和参数中的名称对应
说明:
- 如果返回的是一个
对象
,那框架自动将对象转为json
返回 - 如果返回的字符串中有html元素,那框架自动将字符串解析为html返回
5.4 @RequestParam参数重命名
当用户传递的参数名称与后端方法的形参名称不对应时,我们可以对后端参数重命名
使用@RequestParam注解进行重命名:
@RequestMapping("/login")
public String login(@RequestParam("username") String name,@RequestParam("password") String word){
return name+word;
}
发现名称对应不上也可以获取到,设置了
@RequestParam("username") name
后,就说明从前端拿到参数username然后赋值到name
上,此时前端如果参数名称为name,那此时也是获取不到的
但是此时加上@RequestParam
后,就代表此参数是必须传递
的,如果有参数没有传递的话,会报错
此时呢,为了避免少传参数报错,可以设置@RequestParam的的一个属性required
,将其设置为false
来代表可以不必须传递此参数
@RequestMapping("/login")
public String login(@RequestParam("username") String name,@RequestParam(value = "password",required = false) String word){
return name+word;
}
5.5 @ResquestBody接收JSON对象
先看使用普通方式能否接收到json格式的数据
@RequestMapping("/login")
public Object login(User user){
return user;
}
发现没有获取到json格式的数据
在参数前加上@RequestBody
@RequestMapping("/login")
public Object login(@RequestBody User user){
return user;
}
发现获取到json格式的数据了
5.6 @PathVariable获取URL中参数
此时获取到URL中的参数是获取到带层次的资源路径
,而不是查询字符串
,前面获取的参数都属查询字符串
此时@RequestMapping的映射地址有所改变
//获取带层次的资源路径
@RequestMapping("/login/{username}/{password}")
public String login(@PathVariable String username,@PathVariable String password){
return username+" "+password;
}
注意:加上@PathVariable后,参数也是必传的,同样也可设置required参数为false来保证非必传
@RequestMapping("/login/{username}/{password}")
public String login(@PathVariable String username,@PathVariable(required = false) String password){
return username+" "+password;
}
5.7 @RequestPart上传文件
接收文件类型,必须使用MultipartFile
类型,该类提供了tranferTo
方法(传输),可直接将文件保存在本地路径
//上传文件
@RequestMapping("/getFile")
public void getFile(@RequestPart("file") MultipartFile file) throws IOException {
file.transferTo(new File("D:\photo\file.jpg"));
}
可以在配置文件中设置上传文件的最大大小,防止文件过大上传失败
# 设置最大文件的大小,默认为1MB
spring.servlet.multipart.max-file-size=50MB
# 设置最大请求大小,默认为10MB
spring.servlet.multipart.max-request-size=50MB
如何创建唯一的文件保存路径?
@RequestMapping("/getFile")
public void getFile(@RequestPart("file") MultipartFile file) throws IOException {
//获取上传的文件名称
String fileName = file.getOriginalFilename();
//获取文件后缀名 .xxx
String suffix = fileName.substring(fileName.lastIndexOf("."));
//创建唯一标识
UUID uuid = UUID.randomUUID();
fileName = uuid+suffix;
file.transferTo(new File("D:\photo\"+fileName));
}
5.8 获取Request和Response对象
Request和Response已经内置在框架中,我们可以直接在方法的参数中用,我们可以使用这两个对象像在Servlet中一样,可以干许多事
//获取Request和Response
@RequestMapping("/getReqResp")
public void getReqResp(HttpServletRequest req,HttpServletResponse resp){
Cookie[] cookies = req.getCookies();
String userAgent = req.getHeader("User-Agent");
}
5.9 @RequestHeader获取Header
@RequestMapping("/header")
public String getHeader(@RequestHeader(value = "User-Agent",required = false) String userAgent){
return userAgent;
}
此时也可以设置required属性为false为非必传,否则必须传递,如果没有找到名为User-Agent的header,那程序会报错
5.10 @CookieValue获取Cookie
使用@CookieValue获取指定的Cookie
@RequestMapping("/getCookie")
public Object getCookie(@CookieValue(value = "user",required = false) String cookie){
return cookie;
}
此时也可以设置required属性为false为非必传,否则必须传递,如果没有找到名为user的cookie,那程序会报错
5.11 Session的获取与存储
Session的存储
Session的存储只能使用Request对象来存储
//存储Session
@RequestMapping("/setSession")
public String setSession(HttpServletRequest req){
HttpSession session = req.getSession(true);//没有就创建session
session.setAttribute("user","张三");
return "set success!";
}
Session的获取
使用Request对象获取
//获取Session
@RequestMapping("/getSession")
public String getSession(HttpServletRequest req){
HttpSession session = req.getSession(false);
String username = (String)session.getAttribute("user");
return username;
}
使用@SessionAttribute来获取Session
//获取Session
@RequestMapping("/getSession")
public String getSession(@SessionAttribute(value = "user",required = false) String username){
return username;
}
此时也可以设置required属性为false为非必传,否则必须传递,如果没有找到名为user的session,那程序会报错
6. 返回数据
6.1 @ResponseBody返回数据
Spring MVC默认返回的是视图,也就是静态页面(xxx.html),但是现在都是前后端分离的项目,只需返回前端数据即可,故加上 @ResponseBody
表示返回的是数据
- 加上@ResponseBody后,如果返回的是一个对象,如Map,User,Student…,此时框架会自动将对象转换为JSON返回(相当于返回类型为
application/json
) - 加上@ResponseBody后,如果返回的字符中带有html元素,如 “<h1>哈哈</h1>”,此时框架会自动将该字符串转换为html返回(相当于返回类型为text/html)
返回JSON对象:
@RequestMapping("/test")
@ResponseBody
public Object test(){
Map<String,Object> m = new HashMap<>();
m.put("username","张三");
m.put("age",18);
return m;
}
返回text/html:
@RequestMapping("/test")
@ResponseBody
public String test(){
return "<h1>哈哈</h1>";
}
6.2 请求转发与请求重定向
forward
:请求转发redirect
:请求重定向
请求转发:
@RequestMapping("/forward")
public String forward(){
return "forward:/index.html";
}
请求重定向:
@RequestMapping("/redirect")
public String redirect(){
return "redirect:/index.html";
}
请求转发与请求重定向的区别
- 请求
转发
是服务端行为
,服务端收到请求后,会将请求转发到目标地址,再将目标地址返回的结果返回给客户端,请求转发一次请求
,浏览器地址栏不改变
- 请求
重定向
是客户端行为
,服务端收到请求后,会返回给客户端一个临时的响应头,响应头中包含了目标地址,此时客户端会重新向目标地址发请求,请求重定向两次请求
,浏览器地址栏发生改变