前言

在使用 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/jsonapplication/xml 等。

1、@RequestParam

请求链接举例(GET/POST):?param1=xxx&param2=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-urlencodedform-data 格式数据的,@RequestBody 用来获取非 application/x-www-form-urlencoded 数据的,比如 application/jsonapplication/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"));
}

希望这篇文章对你有所帮助。博客园持续更新,欢迎关注。

博客园:https://www.cnblogs.com/niceyoo

@RequestParam,@RequestBody,@PathVariable注解还分不清吗?相关推荐

  1. @RequestParam @RequestBody @PathVariable 等参数绑定注解详解

    文章主要讲解request 数据到handler method 参数数据的绑定所用到的注解和什么情形下使用. 简介: handler method 参数绑定常用的注解,我们根据他们处理的Request ...

  2. 线性表、顺序表和链表,你还分不清?

    摘要:其实说实话,可能很多人依然分不清线性表,顺序表,和链表之间的区别和联系! 本文分享自华为云社区<程序员必会自己设计线性表(顺序表.链表)>,原文作者:bigsai. 前言 其实说实话 ...

  3. 传参:@RequestParam @RequestBody @PathVariable的作用

    @PathVariable 当使用@RequestMapping URI template 样式映射时,@PathVariable能使传过来的参数绑定到路由上,这样比较容易写出restful api, ...

  4. @RequestParam @RequestBody @PathVariable 等参数绑定

    注解 请求 说明 @PathVariable 处理request url部分 @RequestMapping(" someUrl/{paramId}") @RequestHeade ...

  5. 修改so_货代、海运操作、船务操作还分不清?船公司SO文件看不懂?

    货代是货物代理(freight forwarding agent)的简称,是指经营受他人委托,为其提供代办运输手续,代提.代发.代运货物服务的业务. 货物代理,有些是中间商就是自己没有船或者飞机的或者 ...

  6. 还分不清 Cookie、Session、Token、JWT?

    点击上方蓝色"方志朋",选择"设为星标" 回复"666"获取独家整理的学习资料! 转自:掘金   作者:秋天不落叶 juejin.im/po ...

  7. 用户都跑了,你却还分不清流失用户和流失率

    PMCAFF(www.pmcaff.com):互联网产品社区,是百度,腾讯,阿里等产品经理的学习交流平台.定期出品深度产品观察,互联产品研究首选. 外包大师(www.waibaodashi.com): ...

  8. 还分不清芯片封装?给你整理好了常用的IC封装,点击收藏!

    大家好,我是张巧龙,今天给大家分享一些常见的IC封装形式大全: 转自公众号:ittbank 声明: 本号对所有原创.转载文章的陈述与观点均保持中立,推送文章仅供读者学习和交流.文章.图片等版权归原作者 ...

  9. 普通索引和唯一索引,难道还分不清

    本文分享自华为云社区<MySQL的普通索引和唯一索引到底什么区别?>,作者: JavaEdge. 1 概念区分 普通索引 V.S 唯一索引 普通索引可重复,唯一索引和主键一样不能重复. 唯 ...

最新文章

  1. Java Arrays.asList注意事项
  2. Java中 synchronized 关键字的理解
  3. word如何一键全选_学会这七个Word小技巧,五分钟完成3小时的工作
  4. gdal进行遥感影像读写_如何使用遥感影像进行矿物勘探
  5. C++笔记-Qt中使用Lambda时[]中的形式
  6. 数据集制作——使用labelimg制作数据集
  7. laravel框架简单总结
  8. 已解决IndentationError: unindent does not match any oute r indentation Level
  9. selenium代码中创建浏览器对象_Sinno_Song_新浪博客
  10. 数据增强,扩充了数据集,增加了模型的泛化能力
  11. Java问题定位技术手册
  12. 点阵屏HCMS-3977驱动
  13. 如何去掉德国Kilu免费空间的广告
  14. Ant编译Java项目-QuickStart
  15. oracle查询用户时候被锁,Oracle用户被锁的原因及解决办法
  16. Java中的集合父亲之collection使用和遍历方式--(单列集合顶级接口)
  17. linux能用airport吗_如何设置AppleAirPort无线路由器
  18. Mysql中外键作用以及Navicat建立外键失败总结
  19. R语言解读自回归模型
  20. 什么东西可以替代触屏笔?四款值得推荐的电容笔分享

热门文章

  1. [html] 你知道微信端的浏览器内核是什么吗?
  2. [html] 举例说明图片懒加载的方案有哪些?
  3. [html] 使用svg画出一个矩形
  4. [vue-element] ElementUI是怎么做表单验证的?在循环里对每个input验证怎么做呢?
  5. 工作235:splice
  6. 前端学习(2255)代码是如何冲突得
  7. spring学习(22):分层架构
  8. 第五十一期:互联网不如国企,去BAT的程序员都是diao丝?
  9. std::packaged_task() ---C++17 并发编程
  10. Mac 下安装配置 Python 开发环境