@RequestParam,@RequestBody,@PathVariable注解还分不清吗?
前言
在使用 SpringMVC 开发时,经常遇到前端传递的各种参数,比如 form 表单,JSON 数据,String[] 数组,再或者是最常见的 String 字符串等等,总之大部分场景都是在标题这三个注解来回切换,所以搞清楚这三个注解,日常开发就可以横着走了。
正文
@RequestParam 和 @RequestBody 都是从 HttpServletRequest request 中取参的,而 @PathVariable 是映射 URI 请求参数中的占位符到目标方法的参数中的,接下来一一举例说明。
希望大家能了解:前端在不明确指出 Content-Type 时,默认为 application/x-www-form-urlencoded
格式,@RequestParam 可以获取 application/x-www-form-urlencoded
以及 application/json
这两种类型的参数,但是 @RequestBody 是用来获取非 application/x-www-form-urlencoded
类型的数据,比如 application/json
、application/xml
等。
1、@RequestParam
请求链接举例(GET/POST):?param1=xxx¶m2=yyy
http://javam4.com/m4detail?id=111&tag=java
后端接收举例:
@RequestMapping(value = "/m4detail", method = {RequestMethod.GET,RequestMethod.POST})
public void m4detail(@RequestParam(value="id", required=true) String isId, @RequestParam String tag) {
System.out.println("isId="+isId);
System.out.println("tag="+tag);
}
首先这种方式无论是 GET 还是 POST 请求,都是可以获取到参数的,举例中特意使用了 @RequestParam 注解的一些参数,具体参数如下:
- defaultValue 如果本次请求没有携带这个参数,或者参数为空,那么就会启用默认值
- name 绑定本次参数的名称,要跟URL上面的一样
- required 这个参数不是必须的,如果为 true,不传参数会报错
- value 跟name一样的作用,是name属性的一个别名
2、@PathVariable
请求链接举例(GET/POST):/{id}
http://javam4.com/m4detail/111?tag=java
后端接收举例:
@RequestMapping(value = "/m4detail/{id}", method = {RequestMethod.GET,RequestMethod.POST})
public void m4detail(@PathVariable String id, @RequestParam String tag) {
System.out.println("id="+id);
System.out.println("tag="+tag);
}
然后有的小伙伴可能会问,你这就接收了一个 {id},那我能接受 2 个参数吗?能。
一个 {xx} 就能对应一个参数,那你的请求链接假如是这样:
http://javam4.com/m4detail/111/java
后端接收方式:
@RequestMapping(value = "/m4detail/{id}/{tag}", method = {RequestMethod.GET,RequestMethod.POST})
public void m4detail(@PathVariable String id, @PathVariable String tag) {
System.out.println("id="+id);
System.out.println("tag="+tag);
}
同样 @PathVariable 也有相应的参数:
- name 绑定参数的名称,默认不传递时,绑定为同名的形参。 赋值但名称不一致时则报错
- value 跟name一样的作用,是name属性的一个别名
- required 这个参数不是必须的,如果为 true,不传参数会报错
总结,使用 @PathVariable 需要注意两点:
- 参数接收类型使用基本类型
- 如果@PathVariable标明参数名称,则参数名称必须和URL中参数名称一致
3、@ReuqestBody(不能用于GET请求)
通常后端与前端的交互大多情况下是 POST 请求,尤其是传递大量参数时,毕竟大量参数暴露在浏览的地址栏还是不怎么优雅的,而在 POST 请求中应用 JSON 串对于 Spring MVC 来说是比较友好的,后端使用 @RequestBody 注解就可以方便的实现 JSON 串到接收参数的数据映射。
说明一下 @RequestBody 为什么不能用用于 GET 请求,RequestBody 顾名思义,是将请求参数设置在请求 Body 中的,也就是请求体,而 GET 请求无请求体。
使用 @RequestBody 需要满足如下条件:
- Content-Type 为 application/json,确保传递是 JSON 数据;
- 参数转化的配置必须统一,否则无法接收数据,比如 json、request 混用等
传递参数举例:(JSON数据)
{
"aaa": {
"id": "759791ec-0175-ff808081",
"title": "我是标题",
"content": "我是内容"
},
"bbb": [
"123456"
],
"ccc": 10
}
后端想要接收这个 JSON 数据有两种方式选择,一种是建立与 JSON 数据与之对应的实体,二是直接使用 Map<string,object> 对象接收。
因为 SpringMVC 会帮我们把符合要求的参数封装进实体对象中,所以在参数比较多的情况下,直接使用对象方式会比较方便。
后端接收举例:(实体举例)
@PostMapping("/save")
public void save(@RequestBody QuestionVo vo) {
System.out.println(vo.getAaa().getId());
}
QuestionVo.java 实体:
public class QuestionVo {
private Question aaa;
private List<String> bbb;
private List<String> ccc;
省略get\set方法...
}
public class Question {
private String id;
private String title;
private String content;
省略get\set方法...
}
在这给大家说一下 @RequestBody 在一个请求中只能用一次,如下是报错的:
@PostMapping("/save")
public void save(@RequestBody QuestionVo vo, @RequestBody String niceyoo) {
System.out.println(vo.getAaa().getId());
}
报错信息:
I/O error while reading input message; nested exception is java.io.IOException: Stream closed
但是 @RequestParam 是支持多个使用的。
总结(一定要看)
1、在 GET 请求中可以使用 @RequestParam,不能使用 @RequestBody,@RequestBody 是用来获取请求体中的参数,因为 GET 请求没有请求体,所以不能使用。
2、在 POST 请求中,可以使用 @RequestBody 和 @RequestParam ,其中 @RequestParam 是用来获取 application/x-www-form-urlencoded
、form-data
格式数据的,@RequestBody 用来获取非 application/x-www-form-urlencoded
数据的,比如 application/json
、application/xml
等。
3、一个方法中,可以同时使用多个 @RequestParam ,但是只能使用一个 @RequestBody,否则会报错。
4、@PathVariable 起到的作用就是 URI 请求参数中的占位符到目标方法参数的映射。
5、前端请求的 Content-Type ,默认值为 application/x-www-form-urlencoded
,在这种格式下,后端直接使用 @RequestParam 就可以直接获取指定的参数,但是一旦前端传递的是 JSON 数据,也就是 Content-Type 的值为 application/json
,那么使用 @RequestParam 是取不到值的,不但取不到值还报错。
JSON 数据如下:
{
"name": "哈哈哈哈"
}
后端接收错误演示:
@PostMapping("/save")
public void save(@RequestParam String name) {
System.out.println(name);
}
报错内容:
Required String parameter 'name' is not present
然后小伙伴就会问,那么使用 @RequestBody 可以直接映射 name 值 '哈哈哈哈' 吗?
答案也是否定的,举例如下:
@PostMapping("/save")
public void save(@RequestBody String name) {
System.out.println(name);
}
打印内容如下:
{
"name": "哈哈哈哈"
}
这样其实是将 String name 当做一个对象,Spirng MVC 直接将值映射到 name 上,所以拿到的值是整个 JSON 数据的全部,而创建实体或者是使用 Map 接收就不会出问题,但显然就这 name 一个字段,创建一个实体对象实属浪费,直接用 Map<string,object> / Map<string,string> 接收就可以了:
@PostMapping("/save")
public void save(@RequestBody Map<String,String> map) {
System.out.println(map.get("name"));
}
希望这篇文章对你有所帮助。博客园持续更新,欢迎关注。
@RequestParam,@RequestBody,@PathVariable注解还分不清吗?的更多相关文章
- 【转】@RequestParam @RequestBody @PathVariable 等参数绑定注解详解
@RequestParam @RequestBody @PathVariable 等参数绑定注解详解 2014-06-02 11:24 23683人阅读 评论(2) 收藏 举报 目录(?)[+] 引言 ...
- 还分不清 Cookie、Session、Token、JWT?一篇文章讲清楚
还分不清 Cookie.Session.Token.JWT?一篇文章讲清楚 转载来源 公众号:前端加加 作者:秋天不落叶 什么是认证(Authentication) 通俗地讲就是验证当前用户的身份,证 ...
- @RequestParam @RequestBody @PathVariable 等参数绑定注解详解
文章主要讲解request 数据到handler method 参数数据的绑定所用到的注解和什么情形下使用. 简介: handler method 参数绑定常用的注解,我们根据他们处理的Request ...
- Spring @RequestParam @RequestBody @PathVariable 等参数绑定注解详解
背景 昨天一个人瞎倒腾spring boot,然后遇到了一点问题,所以把这个问题总结一下. 主要讲解request 数据到handler method 参数数据的绑定,所用到的注解和什么情形下使用. ...
- 11.@RequestParam @RequestBody @PathVariable 等参数绑定注解详解
对@RequestMapping进行地址映射讲解之后,该篇主要讲解request 数据到handler method 参数数据的绑定所用到的注解和什么情形下使用: 简介: handler method ...
- @RequestParam @RequestBody @PathVariable 等参数绑定注解详解(转)
引言: 接上一篇文章,对@RequestMapping进行地址映射讲解之后,该篇主要讲解request 数据到handler method 参数数据的绑定所用到的注解和什么情形下使用: 简介: han ...
- (转)@RequestParam @RequestBody @PathVariable 等参数绑定注解详解
引言: 接上一篇文章,对@RequestMapping进行地址映射讲解之后,该篇主要讲解request 数据到handler method 参数数据的绑定所用到的注解和什么情形下使用: 简介: han ...
- springmvc @RequestParam @RequestBody @PathVariable 等参数绑定注解详解
简介: handler method 参数绑定常用的注解,我们根据他们处理的Request的不同内容部分分为四类:(主要讲解常用类型) A.处理requet uri 部分(这里指uri templat ...
- 转载:@RequestParam @RequestBody @PathVariable 等参数绑定注解详解
转载自:https://blog.csdn.net/walkerjong/article/details/7946109#commentBox 因为写的很好很全,所以转载过来 引言:接上一篇文章, ...
随机推荐
- Java进阶专题(十六) 数据结构与算法的应用(上)
前言 学习算法,我们不需要死记硬背那些冗长复杂的背景知识.底层原理.指令语法--需要做的是领悟算法思想.理解算法对内存空间和性能的影响,以及开动脑筋去寻求解决问题的最佳方案.相比编程领域的其他技术 ...
- ubunutu16.04 更改普通用户权限注销后只有guest身份 没有用户身份
第一次踩进百度经验的坑..... 之前对百度经验百信不疑,现在怀疑人生.. 网上搜了很多,也变得小心翼翼,最后姑且相信,但还是有点出入,以下是我的实践: (1)重启ubuntu系统,长按shift进入 ...
- Spring Security 实战干货:客户端OAuth2授权请求的入口
1. 前言 在Spring Security 实战干货:OAuth2第三方授权初体验一文中我先对OAuth2.0涉及的一些常用概念进行介绍,然后直接通过一个DEMO来让大家切身感受了OAuth2.0第 ...
- 三十二张图告诉你,Jenkins构建Spring Boot 有多简单~
持续原创输出,点击上方蓝字关注我 目录 前言 如何安装Jenkins? 环境准备 开始安装Jenkins 初始化配置 访问首页 输入管理员密码 安装插件 创建管理员 实例配置 配置完成 构建Sprin ...
- nginx下配置php5和php7
用的是lnmp 一键安装的 php5.6版本网上百度Ubuntu安装多版本PHP就行 参考文章原链接:http://blog.csdn.net/21aspnet/article/details/476 ...
- 【QT】跨线程的信号槽(connect函数)
线程的信号槽机制需要开启线程的事件循环机制,即调用QThread::exec()函数开启线程的事件循环. Qt信号-槽连接函数原型如下: bool QObject::connect ( const Q ...
- MobaXterm 连接 VirtualBox 6 虚拟机中的 CentOS 7
1 运行环境 本机系统:Windows 7 虚拟机软件:Oracle VM VirtualBox 6 虚拟机系统:CentOS 7 MobaXterm(安装在本机上) 2 MobaXterm - 远端 ...
- Core WebApi项目快速入门(三):踩坑笔记
目前做公司一个项目,遇到了一些坑.跟大家分享,避免再次采坑. 1. 服务端发布应用报错 在windows server上发布程序报错.系统缺少更新包. https://support.microsof ...
- 这么好?中科图新项目经理教你开发LocaSpace功能
LocaSpace是专注于实景三维数据应用的三维数字地球软件,为开发者提供强大.稳定的SDK服务,花费很少的精力即可在自己产品中集成某项功能. 我们将于2018年7月18日至7月20日举办&quo ...
- 第05组 Alpha冲刺 (3/6)(组长)
.th1 { font-family: 黑体; font-size: 25px; color: rgba(0, 0, 255, 1) } #ka { margin-top: 50px } .aaa11 ...