Spring MVC - 为什么不能一起使用@RequestBody和@RequestParam

时间:2013-10-19 16:56:44

标签: spring spring-mvc post http-post http-request-parameters

将HTTP开发客户端与Post请求和Content-Type application / x-www-form-urlencoded一起使用

1)只有@RequestBody

请求 - localhost:8080 / SpringMVC / welcome 在Body - name = abc

代码 -

@RequestMapping(method = RequestMethod.POST)
public String printWelcome(@RequestBody String body, Model model) {
    model.addAttribute("message", body);
    return "hello";
}

//按预期将正文设为'name = abc'

2)只有@RequestParam

请求 - localhost:8080 / SpringMVC / welcome 在Body - name = abc

代码 -

@RequestMapping(method = RequestMethod.POST)
public String printWelcome(@RequestParam String name, Model model) {
    model.addAttribute("name", name);
    return "hello";
}

//按预期将名称命名为'abc'

3)两者在一起

请求 - localhost:8080 / SpringMVC / welcome 在Body - name = abc

代码 -

@RequestMapping(method = RequestMethod.POST)
public String printWelcome(@RequestBody String body, @RequestParam String name, Model model) {
    model.addAttribute("name", name);
    model.addAttribute("message", body);
    return "hello";
}

// HTTP错误代码400 - 客户端发送的请求在语法上不正确。

4)以上与params位置改变

请求 - localhost:8080 / SpringMVC / welcome 在Body - name = abc

代码 -

@RequestMapping(method = RequestMethod.POST)
public String printWelcome(@RequestParam String name, @RequestBody String body, Model model) {
    model.addAttribute("name", name);
    model.addAttribute("message", body);
    return "hello";
}

//没有错误。名字是'abc'。身体是空的

5)一起但获取类型url参数

请求 - localhost:8080 / SpringMVC / welcome?name = xyz 在Body - name = abc

代码 -

@RequestMapping(method = RequestMethod.POST)
public String printWelcome(@RequestBody String body, @RequestParam String name, Model model) {
    model.addAttribute("name", name);
    model.addAttribute("message", body);
    return "hello";
}

// name是'xyz',body是'name = abc'

6)与5)相同,但参数位置已更改

代码 -

@RequestMapping(method = RequestMethod.POST)
public String printWelcome(@RequestParam String name, @RequestBody String body, Model model) {
    model.addAttribute("name", name);
    model.addAttribute("message", body);
    return "hello";
}

// name ='xyz,abc'body为空

有人可以解释这种行为吗?

4 个答案:

答案 0 :(得分:46)

@RequestBody javadoc状态

  

指示方法参数的注释应绑定到正文   网页请求。

它使用HttpMessageConverter的已注册实例将请求主体反序列化为带注释的参数类型的对象。

@RequestParam

  

注释,指示应绑定方法参数   Web请求参数。

  1. Spring将请求正文绑定到使用@RequestBody注释的参数。

  2. Spring将请求体中的请求参数(url编码参数)绑定到方法参数。 Spring将使用参数的名称,即。 name,用于映射参数。

  3. 按顺序解析参数。首先处理@RequestBody。 Spring会消耗所有HttpServletRequest InputStream。然后,当它尝试解析@RequestParam(默认为required)时,查询字符串中没有请求参数或请求正文的剩余部分,即。没有。所以它失败了,因为处理程序方法无法正确处理请求。

  4. @RequestParam的处理程序首先执行操作,阅读HttpServletRequest InputStream可以映射请求参数的内容,即。整个查询字符串/ url编码的参数。它会这样做并获取映射到参数abc的值name。当@RequestBody的处理程序运行时,请求正文中没有任何内容,因此使用的参数是空字符串。

  5. @RequestBody的处理程序读取正文并将其绑定到参数。然后,@RequestParam的处理程序可以从URL查询字符串中获取请求参数。

  6. @RequestParam的处理程序从正文和URL查询字符串中读取。它通常会将它们放在Map中,但由于参数的类型为String,因此Spring会将Map序列化为逗号分隔值。然后,@RequestBody的处理程序再没有任何东西可以从正文中读取。

答案 1 :(得分:3)

回答这个问题为时已晚,但它可能对新读者有所帮助, 看来版本问题。我使用spring 4.1.4运行了所有这些测试,发现@RequestBody@RequestParam的顺序并不重要。

  1. 与您的结果相同
  2. 与您的结果相同
  3. 提供了body= "name=abc"name = "abc"
  4. 与3相同。
  5. body ="name=abc"name = "xyz,abc"
  6. 与5相同。

答案 2 :(得分:0)

您也可以将@RequestParam默认必需状态更改为false,以便不生成HTTP响应状态代码400。这样您就可以按照自己喜欢的顺序放置注释。

@RequestParam(required = false)String name

答案 3 :(得分:0)

这是因为不是非常直接的Servlet规范。如果您正在使用本机HttpServletRequest实现,则无法同时获取URL编码主体和参数。 Spring做了一些变通办法,这使得它变得更加奇怪和不透明。

在这种情况下,Spring(版本3.2.4)使用getParameterMap()方法中的数据为您重新呈现正文。它混合了GET和POST参数并中断了参数顺序。负责混乱的班级是ServletServerHttpRequest。不幸的是,它无法替换,但类StringHttpMessageConverter可以是。

遗憾的是,清洁解决方案并不简单:

  1. 更换StringHttpMessageConverter。复制/覆盖原始班级调整方法readInternal()
  2. 包装HttpServletRequest覆盖getInputStream()getReader()getParameter*()方法。
  3. 在StringHttpMessageConverter方法#readInternal中必须使用以下代码:

        if (inputMessage instanceof ServletServerHttpRequest) {
            ServletServerHttpRequest oo = (ServletServerHttpRequest)inputMessage;
            input = oo.getServletRequest().getInputStream();
        } else {
            input = inputMessage.getBody();
        }
    

    然后必须在上下文中注册转换器。

    <mvc:annotation-driven>
        <mvc:message-converters register-defaults="true/false">
            <bean class="my-new-converter-class"/>
       </mvc:message-converters>
    </mvc:annotation-driven>
    

    此处描述了第二步:Http Servlet request lose params from POST body after read it once