深入剖析:Java 中 @RequestBody 注解的正确使用

news2024/11/20 9:53:38

在 Spring Web 开发中,@RequestBody 是一个常见而又强大的注解。它能够帮助我们轻松地将 HTTP 请求体中的 JSON 数据转换为 Java 对象,从而减少了繁琐的手工解析操作。但看似简单的注解,背后却隐藏着一些坑点和细节。今天,我们将深入探讨 @RequestBody 的使用方法和最佳实践!


一、@RequestBody 是什么?

官方解释:

@RequestBody 用于将 HTTP 请求体中的 JSON、XML 等数据,自动绑定到方法参数对应的 Java 对象上。

简单来说,它让你的控制器方法更优雅:

  • 不用写复杂的解析逻辑。
  • 自动完成反序列化。

常用场景

  • 前端通过 POST 请求发送 JSON 数据。
  • 后端通过 @RequestBody 将这些数据绑定到 Java 对象中。

二、@RequestBody 的基本使用

1. 基本例子

定义请求数据类
public class User {
    private String name;
    private int age;

    // Getter 和 Setter 方法
    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }
}
定义控制器
import org.springframework.web.bind.annotation.*;

@RestController
@RequestMapping("/users")
public class UserController {

    @PostMapping("/create")
    public String createUser(@RequestBody User user) {
        return "User " + user.getName() + " is " + user.getAge() + " years old!";
    }
}
前端发送的请求
POST /users/create
Content-Type: application/json

{
  "name": "Alice",
  "age": 25
}
后端返回的结果
User Alice is 25 years old!

三、@RequestBody 的重要特性

1. 自动反序列化

Spring 使用 Jackson(默认)或其他 JSON 库(如 Gson)将请求体中的 JSON 数据转换为 Java 对象。

2. 必须提供 Content-Type

前端必须在请求头中指定 Content-Type: application/json,否则 Spring 不知道如何处理。

3. 支持嵌套对象

如果 User 类中嵌套了另一个对象,@RequestBody 同样支持。

嵌套对象示例
public class Address {
    private String city;
    private String street;

    // Getter 和 Setter 方法
}

public class User {
    private String name;
    private int age;
    private Address address;

    // Getter 和 Setter 方法
}

JSON 请求:

{
  "name": "Bob",
  "age": 30,
  "address": {
    "city": "New York",
    "street": "5th Avenue"
  }
}

四、@RequestBody 的坑点与解决方法

1. 请求体为空时报错

如果前端发送了一个空的请求体,后端会抛出以下错误:

400 Bad Request: Required request body is missing
解决方法

可以通过设置 @RequestBody 为可选:

@PostMapping("/create")
public String createUser(@RequestBody(required = false) User user) {
    if (user == null) {
        return "Request body is empty!";
    }
    return "User " + user.getName() + " is " + user.getAge() + " years old!";
}

2. 参数校验

如果用户传递的参数不符合预期(比如 age 为负数),需要进行校验。

解决方法

可以配合 @Valid@Validated 使用:

步骤 1:添加校验注解
import jakarta.validation.constraints.Min;
import jakarta.validation.constraints.NotNull;

public class User {
    @NotNull(message = "Name cannot be null")
    private String name;

    @Min(value = 0, message = "Age must be non-negative")
    private int age;

    // Getter 和 Setter 方法
}
步骤 2:在控制器方法中启用校验
import org.springframework.validation.annotation.Validated;

@RestController
@RequestMapping("/users")
public class UserController {

    @PostMapping("/create")
    public String createUser(@Validated @RequestBody User user) {
        return "User " + user.getName() + " is " + user.getAge() + " years old!";
    }
}
步骤 3:全局处理校验异常

可以定义一个全局异常处理器来统一捕获校验错误:

import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.RestControllerAdvice;

@RestControllerAdvice
public class GlobalExceptionHandler {

    @ExceptionHandler(MethodArgumentNotValidException.class)
    public String handleValidationExceptions(MethodArgumentNotValidException ex) {
        return ex.getBindingResult()
                 .getAllErrors()
                 .get(0)
                 .getDefaultMessage();
    }
}

3. 大文件上传

@RequestBody 不适合用于上传大文件,比如图片或视频。

解决方法

对于大文件,建议使用 @RequestParamMultipartFile

@PostMapping("/upload")
public String uploadFile(@RequestParam("file") MultipartFile file) {
    return "Uploaded: " + file.getOriginalFilename();
}

4. 日志打印问题

如果请求体较大(比如 10MB),Spring 默认不会记录完整的请求内容,这对排查问题不太友好。

解决方法

使用过滤器记录完整请求体:

import org.springframework.stereotype.Component;

import javax.servlet.*;
import javax.servlet.http.HttpServletRequest;
import java.io.IOException;

@Component
public class RequestLoggingFilter implements Filter {

    @Override
    public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain)
            throws IOException, ServletException {
        HttpServletRequest httpServletRequest = (HttpServletRequest) request;
        System.out.println("Request Body: " + httpServletRequest.getReader().lines().reduce("", String::concat));
        chain.doFilter(request, response);
    }
}

五、总结

@RequestBody 的优点

  1. 自动反序列化,减少手动解析逻辑。
  2. 支持嵌套对象,便于处理复杂数据。
  3. 配合校验注解,让参数校验更方便。

使用注意事项

  • 确保前端正确设置 Content-Type
  • 小心空请求体和大文件问题。
  • 配合全局异常处理,让校验逻辑更优雅。

@RequestBody 是 Spring Web 开发中的“瑞士军刀”,用好它可以极大地提升开发效率。但在实际开发中,我们需要注意细节,避免常见坑点,让代码更健壮、更易维护。

你在使用 @RequestBody 时遇到过哪些问题?欢迎留言分享!😊

推荐阅读文章

  • 由 Spring 静态注入引发的一个线上T0级别事故(真的以后得避坑)
  • 如何理解 HTTP 是无状态的,以及它与 Cookie 和 Session 之间的联系
  • HTTP、HTTPS、Cookie 和 Session 之间的关系
  • 什么是 Cookie?简单介绍与使用方法
  • 什么是 Session?如何应用?
  • 使用 Spring 框架构建 MVC 应用程序:初学者教程
  • 有缺陷的 Java 代码:Java 开发人员最常犯的 10 大错误
  • 如何理解应用 Java 多线程与并发编程?
  • 把握Java泛型的艺术:协变、逆变与不可变性一网打尽
  • Java Spring 中常用的 @PostConstruct 注解使用总结
  • 如何理解线程安全这个概念?
  • 理解 Java 桥接方法
  • Spring 整合嵌入式 Tomcat 容器
  • Tomcat 如何加载 SpringMVC 组件
  • “在什么情况下类需要实现 Serializable,什么情况下又不需要(一)?”
  • “避免序列化灾难:掌握实现 Serializable 的真相!(二)”
  • 如何自定义一个自己的 Spring Boot Starter 组件(从入门到实践)
  • 解密 Redis:如何通过 IO 多路复用征服高并发挑战!
  • 线程 vs 虚拟线程:深入理解及区别
  • 深度解读 JDK 8、JDK 11、JDK 17 和 JDK 21 的区别
  • 10大程序员提升代码优雅度的必杀技,瞬间让你成为团队宠儿!
  • “打破重复代码的魔咒:使用 Function 接口在 Java 8 中实现优雅重构!”
  • Java 中消除 If-else 技巧总结
  • 线程池的核心参数配置(仅供参考)
  • 【人工智能】聊聊Transformer,深度学习的一股清流(13)
  • Java 枚举的几个常用技巧,你可以试着用用

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

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

相关文章

VSCode+ESP-IDF开发ESP32-S3-DevKitC-1(2)第一个工程 LED心跳灯

VSCodeESP-IDF开发ESP32-S3-DevKitC-1(2)第一个工程 LED心跳灯 前言1.新建工程2.编写控制LED代码3.LED控制独立成.c和.h文件 前言 实际开发中很多时候我们需要有一个类似心跳灯或运行指示灯的灯以不同的状态闪烁以表示程序的运行状态,所以第…

在ubunto18.04安装node 14.16.0

这里演示安装node 14.16.0,其他版本也一样的安装步骤,需要注意1 1.检查本机服务器类型,有的是x64,有的是ARM,先查清楚是什么类型再进行下载,否则会存在编译出错的问题 bash: /opt/node-v14.16.0-linux-x6…

基于YOLOv8深度学习的智慧农业猪行为检测系统研究与实现(PyQt5界面+数据集+训练代码)

随着智慧农业的快速发展,畜牧业的智能化管理已逐渐成为提高生产效率、提升动物福利、降低运营成本的关键手段之一。在此背景下,畜牧场对动物行为的自动化监测需求日益增长,尤其是在大型养猪场,猪群的日常行为检测对于疾病预防、饲…

独立资源池与共享资源池在云计算中各自的优势

在云计算领域,独立资源池和共享资源池是两种关键的资源管理策略,它们各自具有独特的优势,以适应不同的业务需求和场景。 独立资源池的优势 资源独占性:独立资源池为特定应用或用户提供专属的资源,这意味着资源不会被其…

异或和之和

//暴力做法 枚举每个子区间 O(n^3) //优化1 利用前缀异或和快速求出区间异或和 O(n^2) //优化2 处理位运算的常用方法:拆位法 常用的思想:贡献法思想 下面详见优化2: 1.拆位贡献法 2.实战真题1 题目链接:1.异或和之和 - 蓝桥…

【金融风控项目-07】:业务规则挖掘案例

文章目录 1.规则挖掘简介2 规则挖掘案例2.1 案例背景2.2 规则挖掘流程2.3 特征衍生2.4 训练决策树模型2.5 利用结果划分分组 1.规则挖掘简介 两种常见的风险规避手段: AI模型规则 如何使用规则进行风控 **使用一系列逻辑判断(以往从职人员的经验)**对客户群体进行区…

第8章硬件维护-8.2 可维护性和可靠性验收

8.2 可维护性和可靠性验收 可维护性和可靠性验收非常重要,硬件维护工程师在后端发现问题后,总结成可维护性和可靠性需求,在产品立项的时候与新特性一起进行需求分析,然后经过设计、开发和测试环节,在产品中落地。这些需…

在k8s上部署minio

一、 环境 已部署k8s,支持helm部署 二、添加Minio Helm Chart仓库 helm repo add bitnami https://charts.bitnami.com/bitnami -n your_namespace helm repo update -n your_namespace部署带tls的minio helm install minio-s3 bitnami/minio -n your_namespace…

gtest 框架

基本了解 google提供的一个C测试框架,主要就是简化测试单元的书写,具有高效、灵活可拓展的特点 主要特点 简单易用:gtest 提供了清晰且易于使用的 API,便于开发者快速编写单元测试。丰富的断言支持:gtest 提供了多种断…

机器学习—误差分析

帮助运行诊断的最重要的方法是选择下一步要尝试的内容,提高你的学习算法性能,偏差和方差可能是最重要的想法,然后是错误分析。 假设Mcv500,即有500个交叉验证示例,你的算法错误的分类了100个,错误分析过程…

微知-如何查看BlueField DPU上的内存信息,包括内存主频和位宽?(dmidecode -t memory)

背景 在定位DPU上网卡性能的时候,可能涉及到查看内存的主频、位宽、电压等信息,如何快速查看? 命令 dmidecode -t memory实操 可以看到主频是 3200MT/s,另外还能看到位宽,大小,电压等信息。

【AI系统】AI系统架构的组成

AI 系统组成 如图所示,大致可以将 AI 系统分为以下几个具体的方向: AI 训练与推理框架 AI 框架不仅仅是指如 PyTorch 等训练框架,还包括推理框架。其负责提供用户前端的 AI 编程语言,接口和工具链。负责静态程序分析与计算图构建…

竞赛思享会 | 2024年第十届数维杯国际数学建模挑战赛D题【代码+演示】

Hello,这里是Easy数模!以下idea仅供参考,无偿分享! 题目背景 本题旨在通过对中国特定城市的房产、人口、经济、服务设施等数据进行分析,评估其在应对人口老龄化、负增长趋势和极端气候事件中的韧性与可持续发展能力。…

机器学习基础07

目录 1.逻辑回归 1.1原理 1.2API 2.K-Means 2.1算法过程 2.2API 3.SVM(支持向量机) 3.1算法原理​ 3.2API 1.逻辑回归 逻辑回归(Logistic Regression)是机器学习中的一种分类模型,逻辑回归是一种分类算法。 1.1原理 逻辑回归的输…

基于python的在线投票系统小程序u9t2g.

目录 项目介绍开发技术具体实现截图微信开发者工具介绍技术路线开发语言以及框架介绍python-flask核心代码部分展示python-django核心代码部分展示详细视频演示源码获取 项目介绍 投票系统用户端是基于微信小程序,管理员端是基于web网页端, &#xff0c…

android 使用MediaPlayer实现音乐播放--权限请求

在Android应用中,获取本地音乐文件的权限是实现音乐扫描功能的关键步骤之一。随着Android版本的不断更新,从Android 6.0(API级别23)开始,应用需要动态请求权限,而到了android 13以上需要的权限又做了进一步…

向量数据库FAISS之五:原理(LSH、PQ、HNSW、IVF)

1.Locality Sensitive Hashing (LSH) 使用 Shingling MinHashing 进行查找 左侧是字典,右侧是 LSH。目的是把足够相似的索引放在同一个桶内。 LSH 有很多的版本,很灵活,这里先介绍第一个版本,也是原始版本 Shingling one-hot …

【SQL】E-R模型(实体-联系模型)

目录 一、介绍 1、实体集 定义和性质 属性 E-R图表示 2. 联系集 定义和性质 属性 E-R图表示 一、介绍 实体-联系数据模型(E-R数据模型)被开发来方便数据库的设计,它是通过允许定义代表数据库全局逻辑结构的企业模式&#xf…

SIMCom芯讯通A7680C在线升级:FTP升级成功;http升级腾讯云对象储存的文件失败;http升级私有服务器的文件成功

从事嵌入式单片机的工作算是符合我个人兴趣爱好的,当面对一个新的芯片我即想把芯片尽快搞懂完成项目赚钱,也想着能够把自己遇到的坑和注意事项记录下来,即方便自己后面查阅也可以分享给大家,这是一种冲动,但是这个或许并不是原厂希望的,尽管这样有可能会牺牲一些时间也有哪天原…

如何理解岭回归模型?(python)

1 何为岭回归? 岭参数k不是唯一确定的,其估计的回归系数是一个估计族。 2 何为岭迹分析? 1)定义 2)作用 k值的选取原则: 如下图所示,当kk0时,各回归系数的估计值基本都能相对稳…