【JavaEE进阶】SpringMVC

2023-10-16 15:10:00 浏览数 (2)

一. 简单认识SpringMVC

1. 什么是SpringMVC?

SpringMVC是基于Java的Web应用程序开发框架,它是Spring Framework的一部分。它提供了一种基于模型-视图-控制器(Model-View-Controller,MVC)架构的方式来开发灵活、可扩展的Web应用程序。

SpringMVC框架通过将请求的处理流程划分为三个核心组件来实现MVC架构:

  1. 模型(Model):模型表示应用程序中的数据和业务逻辑。在SpringMVC中,模型可以是POJO(Plain Old Java Object)或者通过集成其他持久化技术(如Hibernate)实现的持久化对象。
  2. 视图(View):视图负责渲染模型的数据,并将其展示给用户。视图可以是JSP(JavaServer Pages)、HTML页面、PDF文档、Excel表格等。
  3. 控制器(Controller):控制器接收用户请求并根据用户的操作调用适当的模型和视图。它负责处理用户输入、验证数据、调用业务逻辑和选择合适的视图来响应用户的请求。

SpringMVC框架通过DispatcherServlet作为前端控制器(Front Controller),协调和管理整个请求处理过程。当客户端发送请求时,DispatcherServlet会根据配置信息将请求委派给相应的处理器(Controller),并将结果发送给适当的视图进行渲染。

SpringMVC提供了许多功能和特性,如请求映射、数据绑定、表单处理、数据验证、文件上传、国际化、拦截器、异常处理等。它还支持使用注解和XML配置来进行灵活的开发和配置。

总而言之,SpringMVC是一个用于开发Web应用程序的轻量级、灵活且功能丰富的框架,它通过MVC架构模式提供了一种优雅的方式来构建现代化的Web应用程序。

2. SpringMVC与MVC的关系

MVC 是⼀种思想,而 Spring MVC 是对 MVC 思想的具体实现。SpringMVC是Spring Framework中用于构建Web应用程序的MVC框架,它实现了MVC设计模式的思想,并提供了一套机制来实现模型、视图和控制器之间的解耦和交互。

现在绝大部分的 Java 项目都是基于 Spring(或 Spring Boot)的,而 Spring 的核心就是 Spring MVC。也就是说 Spring MVC 是 Spring 框架的核心模块,而 Spring Boot 是 Spring 的脚手架.Spring MVC 是⼀切项目的基础,Spring、Spring Boot 项目基本都是基于 Spring MVC 的。

二. SpringMVC

学习SpringMVC只需要掌握以下三个功能:

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

1. SpringMVC创建和连接

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

  1. SpringMVC项目的创建:

2. SpringMVC的简单使用

接下来创建一个UserController类:

代码语言:javascript复制
package com.example.demo.controller;

import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

@RestController
public class UserController {
    @RequestMapping("/sayhi")
    public String sayHi(){
        return "hi SpringMVC!";
    }
}

启动项目,在网页中访问http://localhost:8080/sayhi,结果如下:

代码中各注解的含义:

  • @RestController: 相当于@Controller @ResponseBody;@Controller在Spring框架启动时加载.@ResponseBody表示返回非页面数据.
2.1 @RequestMapping 注解介绍

@RequestMapping: @RequestMapping 是 Spring Web 应⽤程序中最常被用到的注解之⼀,它是⽤来注册接口的路由映射的。

路由映射:所谓的路由映射指的是,当用户访问⼀个 url(地址) 时,将用户的请求对应到程序中某个的某个方法的过程就叫路由映射。

@RequestMapping的使用:

代码语言:javascript复制
package com.example.demo.controller;

import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;

@RequestMapping("/Test")//一级路径
@Controller
@ResponseBody
public class TestRequestMapping {
    @RequestMapping("/testReqMap")//二级路径
    public String doTestMapping(){
        return "doTestMapping!";
    }
}

@RequestMapping既可以修饰类也可以修饰方法.当修饰类和方法时,访问的地址是类 方法.

2.2 @RequestMapping支持的请求类型

打开PostMan,当发送GET请求时:

执行成功! 当发生POST请求时:

执行成功. 我们可以看到,默认情况下使用注解 @RequestMapping 可以接收 GETPOST 请求.

2.3 @GetMapping 和 @PostMapping
  1. 关于GET请求的三种写法: (1)@RequestMapping("/index") (2)@RequestMapping(value = "/index",method = RequestMethod.GET) (3)@GetMapping("/index")
  2. 关于POST请求的三种写法; (1)@RequestMapping(value = "/index",method = RequestMethod.POST) (2)@PostMapping("/index")

3. 获取参数

3.1 传递单个参数

在 Spring MVC 中可以直接用方法中的参数来实现传参,比如以下代码:

代码语言:javascript复制
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>My Simple Web Page</title>
</head>
<body>
<h1>Welcome to My Simple Web Page</h1>

<p>This is a paragraph of text.</p>

<ul>
    <li>Item 1</li>
    <li>Item 2</li>
    <li>Item 3</li>
</ul>
</body>
</html>

Test.html文件放置到resources->static目录下:

代码语言:javascript复制
package com.example.demo.controller;

import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;

@Controller
@RequestMapping("/GetParam")
public class GetParameter {
    @RequestMapping("/m1")
    public Object method1(String name){
        System.out.println("参数 name: " name);
        return "/Test.html";
    }
}

在网页中访问:http://localhost:8080/GetParam/m1?name=flower.可以看到html写的简易网页: 同时我们在url中赋值name=flower

同时我们可以在控制台中看到:

3.2 传递对象

首先创建一个Student类:

代码语言:javascript复制
package com.example.demo.controller;

import lombok.Data;

@Data//组合注解相当于@Getter   @Setter   @ToString
    //  @EqualsAndHashCode  @RequiredArgsConstructor  @NoArgsConstructor
public class Student {
    private String name;
    private int age;
    private int id;
}

再运行一下代码:

代码语言:javascript复制
 @RequestMapping("/m2")
    public Object method_2(Student s1){
        System.out.println("对象中的 name:" s1.getName());
        System.out.println("对象中的 age:" s1.getAge());
        System.out.println("对象中的 id: " s1.getId());
        return "/Test.html";
    }

打开网页访问http://localhost:8080/GetParam/m2,运行结果如下:

同时控制台的信息打印如下:

如果我们在url中加入参数:http://localhost:8080/GetParam/m2?name=huang&age=18&id=1

同时控制台打印信息如下:

如果url为:http://localhost:8080/GetParam/m2?name=huang&id=18&age=1

注意此时参数的顺序变化对控制台打印的信息是没有影响的.当有多个参数时,前后端进行参数匹配时,是以参数的名称进行匹配的,因此参数的位置是不影响后端获取参数的结果.

3.3 表单参数传递/传递多个参数(非对象)
代码语言:javascript复制
@RequestMapping("/m3")
    public Object method_3(String name, String pwd) {
        System.out.println("name 参数:"   name);
        System.out.println("pwd 参数:"   pwd);
        return "/Test.html";
    }

使用Postman访问http://localhost:8080/GetParam/m3?name=huang&age=18

访问成功.同时控制台打印信息为:

3.4 后端参数重命名(后端参数映射)

某些特殊的情况下,前端传递的参数key 和我们后端接收的key可以不一致,比如前端传递了一个time给后端,而后端又是用createtime 字段来接收的,这样就会出现参数接收不到的情况,如果出现这种情况,我们就可以使用@RequestParam来重命名前后端的参数值。

代码语言:javascript复制
@RequestMapping("/m4")
    public Object method_4(@RequestParam("time") String createtime) {
        System.out.println("时间:"   createtime);
        return "/Test.html";
    }

我们可以从控制台看到打印的信息正确,也就是说拿到了time.

但是在Postman中出现了400的情况:

查看原码可以看到,@RequestParam中的required默认为true.

required:必须的意思,默认值为true,因此不传递此参数就会报400的错误。 我们可以修改参数实现非必传参数设置. 修改代码为:

代码语言:javascript复制
@RequestMapping("/m4")
    public Object method_4(@RequestParam(value = "time",required = false) String createtime) {
        System.out.println("时间:"   createtime);
        return "/Test.html";
    }
3.5 @RequestBody 接收JSON对象

使用@RequestBody注解可以简化接收JSON数据的过程,提高开发效率,保障安全性,并且支持复杂对象和可扩展性。它是一种常用的处理JSON数据的方式,在前后端分离的开发模式中广泛使用。

后端代码为:

代码语言:javascript复制
@RequestMapping(value = "/m5", method = RequestMethod.POST)
    public Object method_5(@RequestBody Student s1) {
        System.out.println("Student:"   s1);
        return "redirect:/Test.html";
    }

注意此处要使用redirect.但是在使用JSON格式的POST请求接收对象返回时,通常不会使用重定向。因为重定向主要用于将客户端导航到另一个URL,而在处理POST请求时,我们通常期望在同一个URL下进行数据传递和处理。

控制台中信息打印为:

3.6 获取URL中参数@PathVariable

后端代码:

代码语言:javascript复制
@PostMapping("/m6/{name}/{password}")
    public Object method_6(@PathVariable String name, @PathVariable String password) {
        System.out.println("name:"   name);
        System.out.println("password:"   password);
        return "redirect:/Test.html";
    }

使用Postman访问:http://localhost:8080/GetParam/m6/huang/12345:

注意这种方式通常用于网页搜索时在前面显示的情况.

3.7 上传文件@RequestPart

后端代码:

代码语言:javascript复制
@RequestMapping("/m7")
    public Object upload(@RequestPart("myimg6") MultipartFile file) {
        String fileName = UUID.randomUUID()      // 文件名
                file.getOriginalFilename().substring(   // 文件后缀
                        file.getOriginalFilename().lastIndexOf("."));
        File saveFile = new File("E:\JavaEE\upload\"   fileName);
        try {
            file.transferTo(saveFile);
            return true;
        } catch (IOException e) {
            e.printStackTrace();
        }
        return false;
    }

此处使用了UUID. UUID (Universally Unique Identifier) 是一种标识符,用于唯一地标识信息、实体或资源。它是一个128位长的数字,在计算机系统中广泛使用。UUID 通常表示为32位十六进制数,由连字符分隔为五组,形如:xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx,其中每组为8位、4位、4位、4位和12位十六进制数字。

打开Postman,设置文件myimg6longmao.jpg,发送GET请求:

打开upload文件可以看到:

注: 文件大小有要求, 那么可以通过Spring官方文档查询默认值, 进而在配置文件中修改默认大小. Common Application Properties

3.8 获取Cookie/Session/header

获取 RequestResponse 对象 后端代码:

代码语言:javascript复制
@RequestMapping("/param10")
    @ResponseBody
    public String param10(HttpServletResponse response, HttpServletRequest request) {
        String name = request.getParameter("name");
        // 获取所有 cookie 信息
        Cookie[] cookies = request.getCookies();
        return name   " 你好.";
    }

@CookieValue获取cookie: 后端代码:

代码语言:javascript复制
@RequestMapping("/cookie")
    @ResponseBody
    public String cookie(@CookieValue(value = "bite",required = false) String bite) {
        return "cookie:"   bite;
    }

访问网页:http://localhost:8080/GetParam/cookie,设置cookie

@SessionAttribute获取Session: 后端代码:

代码语言:javascript复制
// 存储 session
    @RequestMapping("/setsess")
    @ResponseBody
    public void setSess(HttpServletRequest request) {
        HttpSession session = request.getSession();
        session.setAttribute(SESSION_KEY, "zhangsan");
    }

    // 获取 session
    @RequestMapping("/getsess")
    @ResponseBody
    public Object getSession(@SessionAttribute(SESSION_KEY) String name) {
        return "session -> "   name;
    }

执行:http://localhost:8080/GetParam/setsess后再执行http://localhost:8080/GetParam/getsess可以看到:

@RequestHeader获取 Header: 后端代码:

代码语言:javascript复制
	@ResponseBody
    @RequestMapping("/gethd")
    public Object getHeader(@RequestHeader("Host") String host) {
        return "header -> "   host;
    }

在代码中执行http://localhost:8080/GetParam/gethd:

4. 返回数据

4.1 返回静态页面

前面我们已经提及到了关于返回静态页面,此处做简单介绍即可: 在resources.static中创建前端页面Test.html:

代码语言:javascript复制
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>My Simple Web Page</title>
</head>
<body>
<h1>Welcome to My Simple Web Page</h1>

<p>This is a paragraph of text.</p>

<ul>
    <li>Item 1</li>
    <li>Item 2</li>
    <li>Item 3</li>
</ul>
</body>
</html>

后端代码:

代码语言:javascript复制
package com.example.demo.controller;

import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;

@Controller
@RequestMapping("/Test")
public class TestController {
    @RequestMapping("/test")
    public Object getTest() {
        return "/Test.html";
    }
}

在浏览器中访问:http://localhost:8080/Test/test

注意代码中的返回路径: 当return中加了/之后就代表是从根路径去找Test.html的. 当不加/的时候, 它是在/test底下去找Test.html的, 即在当前目录底下去找Test.html的. 显然在上面的URL并找不到, 因为Test.html是放在根路径下的.

4.2 返回text/html

后端代码:

代码语言:javascript复制
package com.example.demo.controller;
 
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
 
@RestController
public class OwnerController {
    @RequestMapping("/m7")
    public String method_1(){
        return "<h1>hi, h1.</h1>";
    }
}

通过Fiddler抓包可以看到,返回的是text/html:

4.3返回JSON对象

后端代码:

代码语言:javascript复制
 @RequestMapping("/m2")
    public HashMap<String, String> method_2(){
        HashMap<String, String> map = new HashMap<>();
        map.put("C", "C Value");
        map.put("Java", "Java Value");
        map.put("MySQL", "MySQL Value");
        return map;
    }

运行可以得到:

通过Fiddler抓包可以看到,返回的是JSON对象:

请求转发或请求重定向

代码语言:javascript复制
// TestController下, 类注解@RequestMapping("/Test")
    // 请求转发
    @ResponseBody
    @RequestMapping("/fw")
    public String fw(){
        return "forward:/index.html";
    }

    // 请求重定向
    @ResponseBody
    @RequestMapping("/rd")
    public String rd(){
        return "redirect:/index.html";
    }

请求转发和请求重定向的直观区别:

请求转发: 当输入localhost:8080/Test/fw后, URL不变.

请求重定向: 当输入localhost:8080/Test/rd后, URL变为localhost:8080/Test.html

请求转发(Forward)和请求重定向(Redirect)是在Web应用程序中用于处理请求的两种不同的方式。

请求转发(Forward):

  • 请求转发是服务器内部的操作,客户端并不感知。当服务器接收到一个请求后,它可以将请求转发给另一个资源进行处理,然后将结果返回给客户端。
  • 在请求转发中,客户端只发起了一次请求,URL地址没有变化,因此客户端浏览器的地址栏仍显示原始请求的URL。
  • 请求转发可以在同一个web应用程序的不同组件之间进行,如Servlet之间的转发,或JSP页面与Servlet之间的转发。
  • 请求转发可以共享请求的信息(包括请求参数、请求属性等)给目标资源进行处理。

请求重定向(Redirect):

  • 请求重定向是通过发送特殊的响应给客户端,让客户端重新发起一个新的请求
  • 在请求重定向中,服务器会给客户端返回一个特殊的HTTP响应码(通常是302 Found),并在响应头中包含一个新的URL地址,告诉客户端进行重新定向。
  • 客户端收到响应后,会自动发起一个新的GET请求到新的URL地址上,因此客户端的地址栏会显示新的URL地址。
  • 请求重定向可以用于跳转到不同的应用程序、不同的服务器,甚至是跳转到外部网站。

区别:

  • 请求转发是服务器内部操作,客户端无感知,而请求重定向是通过响应告诉客户端重新发起新的请求。
  • 请求转发只有一次请求,URL地址不变,而请求重定向将产生两次请求,第一次是原始请求,第二次是重定向后的请求。
  • 请求转发可以在同一个Web应用程序内进行,共享请求信息,而请求重定向可以跳转到不同的应用程序、服务器和外部网站。
  • 请求转发更高效,不需要额外的网络传输,而请求重定向可能会有额外的网络开销。

添加热部署

热部署(Hot Deployment)是指在系统运行过程中,无需停机或重新启动,即可更新和部署软件或应用程序的新版本或修改。 热部署则提供了一种更加灵活和高效的方式来部署软件。通过热部署,新的代码、配置文件或资源可以在系统运行时被动态加载,而无需重启整个应用程序。这样可以减少系统停机时间,避免用户的中断,提高系统的可用性和稳定性。 热部署技术通常与现代的应用程序框架和容器化技术结合使用,如Java中的热代码替换(Hot Code Replacement)和Spring Boot的热部署功能。通过热部署,开发人员可以快速地进行代码调试、功能添加或修复,并且不会对正在运行的系统产生较大的影响。

  1. pom.xml中引入依赖:
代码语言:javascript复制
		<dependency>
			<groupId>org.springframework.boot</groupId>
			<artifactId>spring-boot-devtools</artifactId>
			<scope>runtime</scope>
			<optional>true</optional>
		</dependency>
  1. Settings中勾选
  1. Settings->Advanced Settings中选择:
  1. 启动

热部署启动成功:

0 人点赞