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
 */
@Controller

public 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. 请求重定向与直接访问新地址效果⼀直,不存在原来的外部资源不能访问;请求转发服务器端转发有可能造成原外部资源不能访问。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

每日一题——轮转数组

1. 题目描述 给定一个整数数组nums&#xff0c;将数组中的元素向右轮转k个位置&#xff0c;其中k是非负数。 示例1: 输入&#xff1a;nums [1,2,3,4,5,6,7]&#xff0c;k 3 输出&#xff1a;[5,6,7,1,2,3,4] 解释&#xff1a; 向右轮转 1步&#xff1a;[7,1,2,3,4,5,6] 向右…

在Linux下探索MinIO存储服务如何远程上传文件

&#x1f308;个人主页&#xff1a;聆风吟 &#x1f525;系列专栏&#xff1a;网络奇遇记、Cpolar杂谈 &#x1f516;少年有梦不应止于心动&#xff0c;更要付诸行动。 文章目录 &#x1f4cb;前言一. 创建Buckets和Access Keys二. Linux 安装Cpolar三. 创建连接MinIO服务公网地…

LED灯驱动模块加载与卸载代码框架

一. 简介 本文来编写 LED灯驱动模块加载与卸载的代码。 二. LED灯驱动模块加载与卸载代码框架 1. 创建工程 我的驱动代码存放目录&#xff1a; ubuntu系统 /home/wangtian/zhengdian_Linux/Linux_Drivers 目录下。 进入 /home/wangtian/zhengdian_Linux/Linux_Drivers 目…

Java开发框架和中间件面试题(4)

27.如何自定义Spring Boot Starter&#xff1f; 1.实现功能 2.添加Properties 3.添加AutoConfiguration 4.添加spring.factory 在META INF下创建spring.factory文件 6.install 28.为什么需要spring boot maven plugin? spring boot maven plugin 提供了一些像jar一样打包…

优先级队列与仿函数

优先级队列 优先级队列 priority_queue 是一种容器适配器&#xff0c;听起来是队列&#xff0c;其实它的底层数据结构是堆&#xff0c;所谓的优先级为默认越大的数优先级越高&#xff0c;即默认为大堆。 使用方式如下面的代码&#xff1a; #include<iostream> #includ…

做抖店需要保证金吗?总共需要多少资金?具体资金投入如下!

我是电商珠珠 做抖店需要保证金吗&#xff1f;这是很多想要入驻的新手常问的问题。我的回答是&#xff0c;需要&#xff01; 抖店平台之所以设立保证金&#xff0c;就是为了约束商家的行为&#xff0c;避免交易市场出现混乱&#xff0c;给用户一个良好的购物体验。 今天呢&a…

【性能优化】MySql数据库查询优化方案

阅读本文你的收获 了解系统运行效率提升的整体解决思路和方向学会MySQl中进行数据库查询优化的步骤学会看慢查询、执行计划、进行性能分析、调优 一、问题&#xff1a;如果你的系统运行很慢&#xff0c;你有什么解决方案&#xff1f; ​关于这个问题&#xff0c;我们通常首先…

Unity中Shader观察空间推导

文章目录 前言一、本地空间怎么转化到观察空间二、怎么得到观察空间的基向量1、Z轴向量2、假设 观察空间的 Y~假设~ (0,1,0)3、X Y 与 Z 的叉积4、Y X 与 Z 的叉积 三、求 [V~world~]^T^1、求V~world~2、求[V~world~]^T^ 四、求出最后在Unity中使用的公式1、偏移坐标轴2、把…

[每周一更]-(第31期):Mysql安装汇总

写自&#xff1a;20230204 23:25 一. mysql rpm二进制包 rpm -Uvh http://repo.mysql.com/mysql-community-release-el6-5.noarch.rpm yum install mysql-community-server service mysqld start set password password(“123456”)二. mysql yum安装 1、安装查看有没有安装…

企业级“RAS”的数据平台如何炼成?

从“看报表”到“数据分析结果直接投入运营”&#xff0c;数字化正在深入企业经营&#xff0c;数据系统正在成为核心生产系统。相应的&#xff0c;企业对“作业挂了”、“系统崩了”、“算不出来”的容忍度越来越低——只有足够稳定、可靠、专业的数据系统&#xff0c;才能及时…

智能优化算法应用:基于社交网络算法3D无线传感器网络(WSN)覆盖优化 - 附代码

智能优化算法应用&#xff1a;基于社交网络算法3D无线传感器网络(WSN)覆盖优化 - 附代码 文章目录 智能优化算法应用&#xff1a;基于社交网络算法3D无线传感器网络(WSN)覆盖优化 - 附代码1.无线传感网络节点模型2.覆盖数学模型及分析3.社交网络算法4.实验参数设定5.算法结果6.…

el-select 全选

<template><div class"container"><el-selectv-model"choosedList"clearablemultiplecollapse-tagsplaceholder"请选择"change"select_Change"><div style"padding: 0 20px; line-height: 34px">&l…

机器学习算法(11)——集成技术(Boosting——梯度提升)

一、说明 在在这篇文章中&#xff0c;我们学习了另一种称为梯度增强的集成技术。这是我在机器学习算法集成技术文章系列中与bagging一起介绍的一种增强技术。我还讨论了随机森林和 AdaBoost 算法。但在这里我们讨论的是梯度提升&#xff0c;在我们深入研究梯度提升之前&#xf…

Python实现AR协方差结构线性回归模型(GLSAR算法)项目实战

说明&#xff1a;这是一个机器学习实战项目&#xff08;附带数据代码文档视频讲解&#xff09;&#xff0c;如需数据代码文档视频讲解可以直接到文章最后获取。 1.项目背景 GLSAR是具有AR协方差结构的广义最小二乘法线性回归模型。 本项目通过GLSAR回归算法来构建AR协方差结构…

RocketMQ实践:确保消息不丢失与顺序性的高效策略

一、使用RocketMQ如何保证消息不丢失&#xff1f; 这个是在面试时&#xff0c;关于MQ&#xff0c;面试官最喜欢问的问题。这个问题是所有MQ都需要面对的一个共性问 题。大致的解决思路都是一致的&#xff0c;但是针对不同的MQ产品又有不同的解决方案。分析这个问题要从以 下几…

02|用LangChain快速构建基于“易速鲜花”本地知识库的智能问答系统

02&#xff5c;用LangChain快速构建基于“易速鲜花”本地知识库的智能问答系统 项目及实现框架 我们先来整体了解一下这个项目。 项目名称&#xff1a;“易速鲜花”内部员工知识库问答系统。 项目介绍&#xff1a;“易速鲜花”作为一个大型在线鲜花销售平台&#xff0c;有自…

基于java的病房管理系统论文

摘 要 当下&#xff0c;如果还依然使用纸质文档来记录并且管理相关信息&#xff0c;可能会出现很多问题&#xff0c;比如原始文件的丢失&#xff0c;因为采用纸质文档&#xff0c;很容易受潮或者怕火&#xff0c;不容易备份&#xff0c;需要花费大量的人员和资金来管理用纸质文…

Linux多线程

目录 一、Linux线程概念1.1 什么是线程&#xff1f;1.2 线程的优点1.3 线程的缺点1.4 线程异常1.5 线程用途1.6 进程VS线程1.7 关于进程和线程的问题 二、Linux线程控制2.1 POSIX线程库2.2 创建线程2.3 线程ID及进程地址空间布局 三、Linux线程终止四、线程等待4.1 为什么需要线…

MySQL——复合查询

目录 一.基本查询回顾 二. 多表查询 三.自连接 四.子查询 1.单行子查询 2.多行子查询 3.多列子查询 4.在from子句中使用子查询 5.合并查询 一.基本查询回顾 准备数据库&#xff1a; 查询工资高于500或岗位为MANAGER的雇员&#xff0c;同时还要满足他们的姓名首字母为…

stm32项目(14)——基于stm32f103zet6的循迹避障小车

1.功能设计 stm32循迹避障小车&#xff0c;使用超声波测距&#xff0c;使用红外循迹模块追踪黑线&#xff0c;实现循迹功能。此外&#xff0c;还可以检测烟雾、火焰、人体、温湿度。温湿度显示在LCD屏幕上。检测到有人、有火焰、有烟雾时&#xff0c;蜂鸣器报警&#xff01; 功…