基于Springboot用axiospost请求接收字符串参数为null的解决方案

问题

今天在用前端 post 请求后端时发现,由于是以 Json对象的形式传输的,后端用两个字符串形参无法获取到对应的参数值

前端代码如下:

axios.post('http://localhost:8083/test/postParams',{a: '1', b:'2'} ,{'Content-Type': 'application/json'}).then(response => {console.log(response.data);}).catch(error => {console.error('There was an error!', error);
});

后端代码如下:

@RequestMapping("/test")
@RestController
@Slf4j
public class TestController {@PostMapping("/postParams")public void postParams(String a, String b) {log.info(String.valueOf(a));log.info(b);}
}

image-20250123171531186

解决

在网上学习了一下,究其原因是Spring Boot 无法直接将 JSON 字符串转换为一个 String 变量, Spring Boot 需要通过相应的机制,将 JSON 字符串解析成可用的 Java 对象或 Map,在学习了某位前辈的文章后,通过自定义注解的方式解决了问题:

​ 总的思路就是,getRequestBody()将请求的json对象字符串先缓存到cache中,然后将该字符串解析成Json对象,在根据对应的方法形参的名字,将值注入进去。

自定义注解类

/*** @author yamu* @version 1.0* @description: 接收前端传的 包装类数据 或 String 自定义注解* @date 2025/1/13 11:05*/
@Target(ElementType.PARAMETER)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface RequestJson {//参数值(对应的键名)String value() default "";
}

定义@RequestJson的方法形参解析器

/*** @author yamu* @version 1.0* @description: 自定义注解 RequestJson 方法形参解析器* @date 2025/1/13 11:07*/
@Component
@Slf4j
public class RequestJsonMethodArgumentResolver implements HandlerMethodArgumentResolver {public static String cache = "";//缓存请求体@Overridepublic boolean supportsParameter(MethodParameter parameter) {return parameter.hasParameterAnnotation(RequestJson.class);}/*** @description  String 参数注入* @param: parameter* @param: mavContainer* @param: webRequest* @param: binderFactory* @returns Object* @author yamu* @date 2025/1/20 14:33*/@Overridepublic Object resolveArgument(MethodParameter parameter, ModelAndViewContainer mavContainer, NativeWebRequest webRequest, WebDataBinderFactory binderFactory) throws Exception {RequestJson requestJson = parameter.getParameterAnnotation(RequestJson.class);HttpServletRequest request = webRequest.getNativeRequest(HttpServletRequest.class);//未指定映射的键时,默认值为形参名String value = requestJson.value();if (value.isEmpty()) {value = parameter.getParameterName();}JSONObject jsonObject = getRequestBody(request);//遍历完最后一个参数,则清理缓存if (parameter.getMethod().getParameterCount() - 1 <= parameter.getParameterIndex()) {cache = "";}//请求的参数为空,直接返回nullif (jsonObject == null) {return null;}return jsonObject.get(value);}/*** 获取参数列表* @param request* @return*/private JSONObject getRequestBody(HttpServletRequest request) {//cache不为空if (!cache.isEmpty()) {return JSONObject.parseObject(cache);}//字符串拼接成Json字符串StringBuilder sb = new StringBuilder();try {BufferedReader reader = request.getReader();char[] buf = new char[1024];int rd;while ((rd = reader.read(buf)) != -1) {sb.append(buf, 0, rd);}} catch (IOException ex) {log.error(ex.getMessage());}cache = sb.toString();return JSONObject.parseObject(sb.toString());}
}

在WebConfig里注册解析器

@Configuration
@Slf4j
public class WebConfig extends WebMvcConfigurationSupport  {@Autowiredprivate RequestJsonMethodArgumentResolver requestJsonMethodArgumentResolver;/*** @description 配置方法解析器* @param: argumentResolvers* @returns void* @author yamu* @date 2025/1/23 16:00*/@Overridepublic void addArgumentResolvers(List<HandlerMethodArgumentResolver> argumentResolvers) {argumentResolvers.add(requestJsonMethodArgumentResolver);}
}

在方法形参上加上注解

@RequestMapping("/test")
@RestController
@Slf4j
public class TestController {@PostMapping("/postParams")public void postParams(@RequestJson String a, @RequestJson String b) {log.info(a);log.info(b);}
}

image-20250123174734261

上述方式存在几个问题:

  1. 由于要缓存请求的 Json字符串,所以在每次请求完之后要清除cache,上述方法是在方法形参的最后一个并且加了@RequestJson注解的参数才可以清理
  2. 由于需要对每个参数进行赋值,所以需要对每个要注入的参数都要加上@RequestJson注解
  3. 处理包装类或字符串类时,形参类型需要强一致(不能用Stringl类型接收一个Integer的参数值),同时也无法处理复杂的对象类型

后续在逐渐的深入学习后我会优化上述方式。

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

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

相关文章

ios打包:uuid与udid

ios的uuid与udid混乱的网上信息 新人开发ios&#xff0c;发现uuid和udid在网上有很多帖子里是混淆的&#xff0c;比如百度下&#xff0c;就会说&#xff1a; 在iOS中使用UUID&#xff08;通用唯一识别码&#xff09;作为永久签名&#xff0c;通常是指生成一个唯一标识&#xf…

【豆包MarsCode 蛇年编程大作战】蛇形烟花

项目体验地址&#xff1a;项目体验地址 官方活动地址&#xff1a;活动地址 目录 【豆包MarsCode 蛇年编程大作战】蛇形烟花演示 引言 豆包 MarsCode介绍 项目准备 第一步&#xff1a;安装插件 第二步&#xff1a;点击豆包图标来进行使用豆包 使用豆包 MarsCodeAI助手实…

leetcode151-反转字符串中的单词

leetcode 151 思路 时间复杂度&#xff1a;O(n) 空间复杂度&#xff1a;O(n) 首先将字符串转为数组&#xff0c;这样可以方便进行操作&#xff0c;然后定义一个新的数组来存放从后到前的单词&#xff0c;由于arr中转换以后可能会出现有些项是空格的情况&#xff0c;所以需要判…

Precharge和Self-refresh的区别

一、区别说明 操作目标&#xff1a; Precharge&#xff08;预充电&#xff09;&#xff1a;此操作的主要目标是关闭存储器中某个或所有Bank中现有的工作行&#xff08;active row&#xff09;&#xff0c;并准备打开新的工作行。这是为了确保存储器的连续访问能够高效、可靠地进…

理解C++运行时类型识别符: typeid

1. 格式 typeid (type-id) typeid (expression) typeid 运算符允许在运行时确定对象的类型。 typeid 的结果是一个 const type_info&。该值是对 type_info 对象的引用&#xff0c;该对象表示 type-id 或表达式的类型&#xff0c;具体取决于使用哪种形式的 typeid。有关…

miniconda学习笔记

文章主要内容&#xff1a;演示miniconda切换不同python环境&#xff0c;安装python库&#xff0c;使用pycharm配置不同的conda建的python环境 目录 一、miniconda 1. 是什么&#xff1f; 2.安装miniconda 3.基本操作 一、miniconda 1. 是什么&#xff1f; miniconda是一个anac…

中国特色-流程操作概念

一、核心特点 高度灵活性&#xff1a;支持业务用户对流程进行定义和改进&#xff0c;提供基于Web的可视化流程设计器&#xff0c;方便非技术背景的业务人员操作。 支持动态性需求&#xff1a;能够满足临时动态性需求&#xff0c;如任意回退、会签、加签、减签等&#xff0c;这…

基于本地事务表+MQ实现分布式事务

基于本地事务表MQ实现分布式事务 引言1、原理2、本地消息表优缺点3、代码实现3.1、代码执行流程3.2、项目结构3.3、项目源码 引言 本地消息表的方案最初由ebay的工程师提出&#xff0c;核心思想是将分布式事务拆分成本地事务进行处理。本地消息表实现最终一致性。本文主要学习…

HTML<label>标签

例子 三个带标签的单选按钮&#xff1a; <form action"/action_page.php"> <input type"radio" id"html" name"fav_language" value"HTML"> <label for"html">HTML</label><br&…

2025,“鱿鱼游戏”闯入AI赛道

文 | 智能相对论 作者 | 叶远风 “鱿鱼游戏”一词随着同名剧集的火爆而持续走红&#xff0c;在全球范围掀起了广泛热议。而这种无限生存流的游戏模式&#xff0c;既残酷又现实&#xff0c;像极了商业市场的搏杀与淘汰。 1月20日&#xff0c;DeepSeek发布了全新的开源推理大模…

CSS 中的 id 和 class 选择器

在 CSS 中&#xff0c;id 和 class 是两个常用的选择器&#xff0c;它们用于为 HTML 元素添加样式。虽然它们的功能相似&#xff0c;但在使用场景和具体用法上有很大的区别。本文将详细介绍 id 和 class 的区别&#xff0c;并通过实例帮助你更好地理解它们的应用场景&#xff0…

Prometheus+grafana实践:Doris数据库的监控

文章来源&#xff1a;乐维社区 Doris数据库背景 Doris&#xff08;Apache Doris&#xff09;是一个现代化的MPP&#xff08;Massive Parallel Processing&#xff0c;大规模并行处理&#xff09;数据库&#xff0c;主要用于在线分析处理&#xff08;OLAP&#xff09;场景。 D…

【unity游戏开发之InputSystem——02】InputAction的使用介绍(基于unity6开发介绍)

文章目录 前言一、InputAction简介1、InputAction是什么&#xff1f;2、示例 二、监听事件started 、performed 、canceled1、启用输入检测2、操作监听相关3、关键参数 CallbackContext4、结果 三、InputAction参数相关1、点击齿轮1.1 Actions 动作&#xff08;1&#xff09;动…

现代JavaScript网页设计

现代JavaScript网页设计&#xff1a;打造沉浸式3D粒子交互系统 案例概述 本文将实现一个基于WebGL的3D粒子交互系统&#xff0c;结合物理引擎与光线追踪技术&#xff0c;创造出具有以下高级特性的现代网页体验&#xff1a; 动态粒子矩阵&#xff08;100,000粒子实时渲染&…

基于 WEB 开发的在线学习系统设计与开发

标题:基于 WEB 开发的在线学习系统设计与开发 内容:1.摘要 摘要&#xff1a;随着互联网技术的飞速发展&#xff0c;在线学习已经成为一种重要的学习方式。本文介绍了基于 WEB 开发的在线学习系统的设计与开发过程。该系统采用了 B/S 架构&#xff0c;使用 Java 语言和 MySQL 数…

从Spring请求处理到分层架构与IOC:注解详解与演进实战

引言 在Spring开发中&#xff0c;请求参数处理、统一响应格式、分层架构设计以及依赖管理是构建可维护应用的核心要素。然而&#xff0c;许多开发者在实践中常面临以下问题&#xff1a; 如何规范接收不同格式的请求参数&#xff1f; 为何要引入分层架构&#xff1f; 什么是控…

逐笔成交逐笔委托Level2高频数据下载和分析:20250124

逐笔成交逐笔委托下载 链接: https://pan.baidu.com/s/1UWVY11Q1IOfME9itDN5aZA?pwdhgeg 提取码: hgeg Level2逐笔成交逐笔委托数据分享下载 通过Level2逐笔成交与逐笔委托的详细数据&#xff0c;这种以毫秒为单位的信息能揭示许多关键点&#xff0c;如庄家意图、误导性行为&…

详解Redis的Zset类型及相关命令

目录 Zset简介 ZADD ZCARD ZCOUNT ZRANGE ZREVRANGE ZRANGEBYSCORE ZPOPMAX BZPOPMAX ZPOPMIN BZPOPMIN ZRANK ZREVRANK ZSCORE ZREM ZREMRANGEBYRANK ZREMRANGEBYSCORE ZINCRBY ZINTERSTORE 内部编码 应用场景 Zset简介 有序集合相对于字符串、列表、哈希…

深度解析:哪种心磁图技术是心脏检查的精准之选?

在全球心血管疾病的阴影日益笼罩的今天&#xff0c;医学界正积极寻求一种无损、无创、无辐射的心脏健康监测方式。心磁图仪&#xff08;MCG&#xff09;&#xff0c;这一前沿技术&#xff0c;凭借其独特的优势&#xff0c;悄然成为心脏电磁功能监测的新星。它不仅为心肌缺血、心…

jupyter配置说明

使用以下命令修改jupyter的配置文件参数&#xff1a; vim /root/.jupyter/jupyter_lab_config.py #这里填写远程访问的IP名&#xff0c;填*则默认是主机IP名 c.ServerApp.ip * # 这里的密码填写上面生成的密钥 c.ServerApp.password ************************************…