使用FeignClient调用POST表单Body内没有参数问题

 更新时间:2022年03月22日 08:38:12   作者:五岁程序员  
这篇文章主要介绍了使用FeignClient调用POST表单Body内没有参数问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

FeignClient调用POST表单Body没有参数

1、问题:在使用FeignClient调用第三方接口,对方是普通POST接口,但是调用返回400,后发现请求体没有携带需要的参数。

2、查找资料发现,FeignClient默认POST请求发送参数为JSON格式,如需要使用表单形式,必须自定义设置。

3、附代码

import com.lenovo.mt.config.FormFeignConfiguration;
import com.lenovo.mt.remote.eservice.vo.EServiceToken;
import org.springframework.cloud.openfeign.FeignClient;
import org.springframework.http.MediaType;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestParam;
/**
 * @Description
 * @Author zhaoke
 * @Date 2019/12/16 10:42
 **/
@FeignClient(name = "***", url = "${***}",configuration = FormFeignConfiguration.class, fallback = TokenFail.class)
public interface ITokenRemote {
    @PostMapping(value = "***",consumes = MediaType.APPLICATION_FORM_URLENCODED_VALUE)
    EServiceToken getToken(@RequestParam("client_id")String clientId,@RequestParam("client_secret")String clientSecret,@RequestParam("grant_type")String grantType);
}

其中主要设置在FormFeignConfiguration这个类中

import feign.codec.Encoder;
import feign.form.FormEncoder;
import org.springframework.beans.factory.ObjectFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.autoconfigure.http.HttpMessageConverters;
import org.springframework.cloud.openfeign.support.SpringEncoder;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Primary;
import org.springframework.context.annotation.Scope;
import static org.springframework.beans.factory.config.BeanDefinition.SCOPE_PROTOTYPE;
/**
 * @Description
 * @Author zhaoke
 * @Date 2019/12/17 13:05
 **/
@Configuration
public class FormFeignConfiguration {
    @Autowired
    private ObjectFactory<HttpMessageConverters> messageConverters;
    @Bean
    @Primary
    @Scope(SCOPE_PROTOTYPE)
    Encoder feignFormEncoder() {
        return new FormEncoder(new SpringEncoder(this.messageConverters));
    }
}

FeignClient参数问题

使用Get方式调用服务提供者

服务消费者,需要在所有参数前加上@RequestParam注解

服务消费者,指明是Get方式,如果不指明那么在满足条件1 的情况下默认使用get方式

1和2都满足才使用Get方式

使用Post方式调用服务提供者

在所有参数前面加上@RequestParam注解,并且指明使用的Post方式

在参数前面加上@RequestBody注解(有且只有一个@ RequestBody)或者什么也没有(有多个参数其余参数前面都要加上@RequestParam)

1和2满足一个就使用Post方式

三种情况(无参、单参、多参)

无参Get

如果发送无参的Get请求,可以不指定method,默认就是Get请求

无参Post

指定method是Post方式,否则就是Get请求

单参Get

指定该参数是@RequestParam修饰,这时候method默认是Get,也可以自己指定Get方式

单参Post

该参数前面加@RequestParam修饰,并且使用Post方式

或者

使用@RequestBody修饰该参数,或者没使用@RequestParam和@RequestBody修饰该参数,再这个使用不论指定Post还是Get都会使用Post进行请求

注意:

在使用fegin消费服务的时候,如果参数前面什么也没写,默认是@RequestBody来修饰的

@RequestBody修饰的参数需要以Post方式来传递

在服务提供者的Controller中:

1.如果要接收@RequestBody指明的参数,那么对应方法的对应参数前一定要有@RequestBody;(如果没有的话,收到的参数值就为null;如果写成@RequestParam的话,那么feign调用会失败)

2.如果接收@ RequestParam指明的参数,那么可以写@ RequestParam也可以不写,如果参数名称一致不用,不一致需要写

多参Get

需要在所有参数前面添加@ RequestParam注解,这时候可以加Get也可以不加Get

多参Post

服务消费者中,最多只能有一个参数是@RequestBody指明的,其余的参数必须使用@RequestParam指明

也可以在第一个参数前面什么也不写(默认使用@RequestBody修饰)其余的参数都需要加@RequestParam

如果所有参数都是以@RequestParam修饰,并且使用的是Post方式,那么也是以Post方式来请求

在服务提供者的Controller中:

1.如果服务消费者传过来参数时,全都用的是@RequestParam的话,那么服务提供者的Controller中对应参数前可以写@RequestParam,也可以不写(当两边参数名字一致时,可以省略不写)

2. 如果服务消费者传过来参数时,有@RequestBody的话,那么服务提供者的Controller中对应参数前必须要写@RequestBody(如果是多参数的话,其余参数前视情况可以写@RequestParam,也可以不写)

@RequestBody和@RequestParam的区别

@RequestParam用来处理Content-Type: 为 application/x-www-form-urlencoded编码的内容,在Http中如果不指定Content-Type则默认使用application/x-www-form-urlencoded类型

@ RequestBody,处理HttpEntity传递过来的数据,用来处理非Content-Type: application/x-www-form-urlencoded编码格式的数据

Get请求没有HttpEntity所以不用@ RequestBody

POST请求中,通过HttpEntity传递的参数,必须要在请求头中声明数据的类型Content-Type,SpringMVC通过使用HandlerAdapter 配置的HttpMessageConverters来解析HttpEntity中的数据,然后绑定到相应的bean上

总结:Get不能使用表单,只能在url中传参,传参方式只有这一种。 Post可以使用表单,也可以在url中传参。使用表单时有几种数据类型(表现为数据的存储位置不同):

1、x-www-form-urlencoded 参数存储在query中 用@RequestParam接收。

2、formdata 参数存储在body中,用@RequestBody接收,文件类型用@RequestPart接收。

3、raw(josn,xml) 参数存储在body中 用@RequetBody接收。

总结一下: 凡是放在body中的都可以用@RequestBody接收,文件类型的数据可以用@RequestPart接收。 凡是放在query中的都可以用@RequestParam接收,包括Get方式提交和Post(x-www-form-urlencoded)方式提交的

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • 详谈Lock与synchronized 的区别

    详谈Lock与synchronized 的区别

    下面小编就为大家带来一篇详谈Lock与synchronized 的区别。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-05-05
  • Sharding JDBC读写分离实现原理及实例

    Sharding JDBC读写分离实现原理及实例

    这篇文章主要介绍了Sharding JDBC读写分离实现原理及实例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-12-12
  • Java中String类getBytes()方法详解与完整实例

    Java中String类getBytes()方法详解与完整实例

    这篇文章主要给大家介绍了关于Java中String类getBytes()方法详解与完整实例的相关资料,getBytes()是Java编程语言中将一个字符串转化为一个字节数组byte[]的方法,需要的朋友可以参考下
    2023-10-10
  • Java长度不足左位补0的3种实现方法

    Java长度不足左位补0的3种实现方法

    这篇文章主要介绍了Java长度不足左位补0的3种实现方法小结,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-12-12
  • Spring如何利用@Value注解读取yml中的map配置

    Spring如何利用@Value注解读取yml中的map配置

    这篇文章主要介绍了Spring如何利用@Value注解读取yml中的map配置,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-02-02
  • vue数据响应式原理重写函数实现数组响应式监听

    vue数据响应式原理重写函数实现数组响应式监听

    Vue的通过数据劫持的方式实现数据的双向绑定,即使用Object.defineProperty()来实现对属性的劫持,但是Object.defineProperty()中的setter是无法直接实现数组中值的改变的劫持行为的,需要的朋友可以参考下
    2023-05-05
  • java web上传文件和下载文件代码片段分享

    java web上传文件和下载文件代码片段分享

    这篇文章主要为大家详细介绍了java web上传文件和下载文件代码片段,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-01-01
  • SpringBoot 基于 MongoTemplate 的工具类过程详解

    SpringBoot 基于 MongoTemplate 的工具类过程详解

    MongoDB是一个高性能,开源,无模式的文档型数据库,是当前NoSql数据库中比较热门的一种,这篇文章主要介绍了SpringBoot基于MongoTemplate的工具类,需要的朋友可以参考下
    2023-09-09
  • 详解SpringBoot中@PostMapping注解的用法

    详解SpringBoot中@PostMapping注解的用法

    在SpringBoot中,我们经常需要编写RESTful Web服务,以便于客户端与服务器之间的通信,@PostMapping注解可以让我们更方便地编写POST请求处理方法,在本文中,我们将介绍@PostMapping注解的作用、原理,以及如何在SpringBoot应用程序中使用它
    2023-06-06
  • spring boot2.0图片上传至本地或服务器并配置虚拟路径的方法

    spring boot2.0图片上传至本地或服务器并配置虚拟路径的方法

    最近写了关于图片上传至本地文件夹或服务器,上传路径到数据库,并在上传时预览图片。本文通过实例代码给大家分享spring boot2.0图片上传至本地或服务器并配置虚拟路径的方法,需要的朋友参考下
    2018-12-12

最新评论