服务器之家:专注于服务器技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - Java教程 - Feign的一个注解居然隐藏这么多知识!

Feign的一个注解居然隐藏这么多知识!

2021-09-28 22:54java金融 Java教程

最近由于业务的需要,需要接入下阿里云的一个接口,打开文档看了看这个接口看下来还是比简单的目测个把小时就可以搞定,但是接入的过程还是比较坎坷的。

Feign的一个注解居然隐藏这么多知识!

引言

最近由于业务的需要,需要接入下阿里云的一个接口,打开文档看了看这个接口看下来还是比简单的目测个把小时就可以搞定,但是接入的过程还是比较坎坷的。首先我看了看他给的示例,首先把阿里云文档推荐的demo下载下来,把它的例子跑起来,替换下几个必要的参数比如秘钥啥的。这些秘钥一般公司都会有专职的人员与阿里云去对接,你只要负责管他要就行了。

不过也不排除也有得公司需要自己去对接阿里云。说到这里就想吐槽下,对接阿里云的时候技术支持群居然是钉钉,所以需要他们的支持就必须要下载个钉钉, 电脑上莫名的有需要多装一个软件。扯远了我们还是回到正题,把它demo下载下来,然后把对应的秘钥等参数替换下,然后运行下demo看看是否能够正常返回结果,做这一步主要是为了保证产品给过来的秘钥等参数是否正确。如果能够掉通接口,那就说明参数没啥问题的接着我们就可以着手来写业务代码了。

接入阿里云二要素认证https://market.aliyun.com/products/57000002/cmapi029454.html?spm=5176.10695662.1194487.1.60066c190NsSkZ#sku=yuncode2345400003 把官网的demo下载下来跑起来看看,官网给出的例子还是比较简单粗暴的,就是封装了一个Apache的httplcient工具类一大坨的代码,个人还是习惯性的使用feign来进行调用,因为feign的代码干净整洁,虽然底层也是通过HttpClient来实现,但是实现对我来说是无感的,毕竟业务代码看起来干净整洁。它的demo如下:

  1. public static void main(String[] args) { 
  2.      String host = "https://safrvcert.market.alicloudapi.com"
  3.      String path = "/safrv_2meta_id_name/"
  4.      String method = "GET"
  5.      String appcode = "你自己的AppCode"
  6.      Map<String, String> headers = new HashMap<String, String>(); 
  7.      //最后在header中的格式(中间是英文空格)为Authorization:APPCODE 83359fd73fe94948385f570e3c139105 
  8.      headers.put("Authorization""APPCODE " + appcode); 
  9.      Map<String, String> querys = new HashMap<String, String>(); 
  10.      querys.put("__userId""__userId"); 
  11.      querys.put("customerID""customerID"); 
  12.      querys.put("identifyNum""identifyNum"); 
  13.             querys.put("identifyNumMd5""identifyNumMd5"); 
  14.      querys.put("userName""userName"); 
  15.      querys.put("verifyKey""verifyKey"); 
  16.  
  17.  
  18.      try { 
  19.       /** 
  20.       * 重要提示如下: 
  21.       * HttpUtils请从 
  22.       * https://github.com/aliyun/api-gateway-demo-sign-java/blob/master/src/main/java/com/aliyun/api/gateway/demo/util/HttpUtils.java 
  23.       * 下载 
  24.       * 
  25.       * 相应的依赖请参照 
  26.       * https://github.com/aliyun/api-gateway-demo-sign-java/blob/master/pom.xml 
  27.       */ 
  28.       HttpResponse response = HttpUtils.doGet(host, path, method, headers, querys); 
  29.       //错误信息见X-Ca-Error-Message字段 
  30.                 System.out.println(response.toString()); 
  31.       //获取response的body 
  32.       System.out.println(EntityUtils.toString(response.getEntity())); 
  33.      } catch (Exception e) { 
  34.       e.printStackTrace(); 
  35.      } 
  36.  } 
  1. HttpResponse response = HttpUtils.doGet(host, path, method, headers, querys); 

根据它提供的代码我们可以看出来他是用一个httpUtils 类来实现http请求的,我们可以把这个httpClient类 替换成我们的FeignClient替换后的代码如下:

  1. @FeignClient(name = "verifyIdCardAndNameFeignClient", url = "https://safrvcert.market.alicloudapi.com"
  2. public interface VerifyIdCardAndNameFeignClient { 
  3.     @RequestMapping(value = "/safrv_2meta_id_name/", method = RequestMethod.GET, produces = MediaType.APPLICATION_JSON_UTF8_VALUE) 
  4.     Response verifyIdCardAndNameMap(@RequestParam Map<String,String> app, @RequestHeader("Authorization") String authorization); 

相对比较下来下面这个HttpClientUtils代码是不是比较简洁

Feign的一个注解居然隐藏这么多知识!

按照这个demo功能确实是实现了,说实话个人还是不是很喜欢用map来作为参数,map作为入参的话,参数全靠猜可读性以及可维护性有点差,个人还是习惯性的封装一个javaBean作为实体。阿里文档其实也有提到一嘴,虽然他只说到数据查询这一层。

Feign的一个注解居然隐藏这么多知识!

下面我们就修改下请求参数把它改成一个javaBean,改变后的代码

  1. @RequestMapping(value = "/safrv_2meta_id_name/", method = RequestMethod.GET, produces = MediaType.APPLICATION_JSON_UTF8_VALUE) 
  2. Response verifyIdCardAndNameDTO(@RequestBody AliyunVerifyIdCardAndNameReq app, @RequestHeader("Authorization") String authorization); 

Feign的一个注解居然隐藏这么多知识!

请求并没有成功,根据报错返回的信息看下来应该是没有接受到参数。我们是GET请求的方式然后参数传递的是实体导致没有接收到。feignClient不支持get方式传递实体类吗?后来经过查询资料发现了一个注解@SpringQueryMap 我们把上述代码@RequestBody替换成@SpringQueryMap完美解决这个问题

@SpringQueryMap

在spring cloud 2.1.x 以上的版本,提供了一个新的注解@SpringQueryMap,为何这个注解可以帮我们实现。源码之下无秘密,我们可以翻翻feign的源码相对来说应该是比较简单的,我们可以简单的来看下源码。看源码是不是也不知道从哪里看起,从头看到尾肯定也不现实, 不从头开始看,又不知道源码在哪里,有个很简单的方法我们直接拿着这个注解全局搜一下,看看有哪些地方使用到了,在每个地方都打上一个断点试试

Feign的一个注解居然隐藏这么多知识!

我们全局搜下发现使用的地方主要在QueryMapParameterProcessor这个类里面。所以我们可以在这个类里面打上一个断点试试。

  1. /** 
  2.  * {@link SpringQueryMap} parameter processor. 
  3.  * 
  4.  * @author Aram Peres 
  5.  * @see AnnotatedParameterProcessor 
  6.  */ 
  7. public class QueryMapParameterProcessor implements AnnotatedParameterProcessor { 
  8.  
  9.  private static final Class<SpringQueryMap> ANNOTATION = SpringQueryMap.class; 
  10.  
  11.  @Override 
  12.  public Class<? extends Annotation> getAnnotationType() { 
  13.   return ANNOTATION; 
  14.  } 
  15.  
  16.  @Override 
  17.  public boolean processArgument(AnnotatedParameterContext context, Annotation annotation, Method method) { 
  18.   int paramIndex = context.getParameterIndex(); 
  19.   MethodMetadata metadata = context.getMethodMetadata(); 
  20.   if (metadata.queryMapIndex() == null) { 
  21.    metadata.queryMapIndex(paramIndex); 
  22.    metadata.queryMapEncoded(SpringQueryMap.class.cast(annotation).encoded()); 
  23.   } 
  24.   return true
  25.  } 

我们发现打这个类的话在容器启动的时候会进行加载,并且会执行processArgument方法,这个我们先不管这个方法,接下来我们来看看 Feign真正发起调用的地方找到SynchronousMethodHandler#invoke方法

  1. public RequestTemplate create(Object[] argv) { 
  2.  ... 省略部分代码 
  3.  // metadata.queryMapIndex() 就是QueryMapParameterProcessor #processArgument方法赋值的 
  4.       if (metadata.queryMapIndex() != null) { 
  5.         // add query map parameters after initial resolve so that they take 
  6.         // precedence over any predefined values 
  7.         // 通过下标获取到需要特殊处理的对象,这里有个问题只会处理方法参数的第一个@SpringQueryMap注解, 
  8.         // 原因就是QueryMapParameterProcessor #processArgument这个方法只会把第一个下标赋值进去,然后这里也只会取第一个下标,所以只会处理第一个@SpringQueryMap注解 
  9.         Object value = argv[metadata.queryMapIndex()]; 
  10.         //将对象转换为map  这里需要注意下默认使用解析参数的是FieldQueryMapEncoder类所以它并不会去解析父类的参数,如果需要解析父类的参数我们需要在feign的Config里面指定QueryMapEncoder为FieldQueryMapEncoder 
  11.         Map<String, Object> queryMap = toQueryMap(value); 
  12.         //拼接解析完成的对象为URL参数 
  13.         template = addQueryMapQueryParameters(queryMap, template); 
  14.       } 
  15. ... 省略部分代码 

上述代码逻辑还是挺好理解的

  • 首先去判断是否需要处理下querymap
  • 通过下标获取到需要特殊处理的对象
  • 将对象转换为map(这里有个坑默认不会去解析父类的字段)
  • 拼接追加map到url中

总结

上面通过@SpringQueryMap注解实现了get传参,但是如果需要传递多个@SpringQueryMap注解我们可以怎么来实现呢?

或者我们可以自己动手来实现一个我们自己的SpringQueryMap,我们该如何实现?

@SpringQueryMap注解默认是不会去解析父类的参数,如果需要解析父类的参数需要修改Feign的config# QueryMapEncoder为FieldQueryMapEncoder。

如果我们自己去实现了一个AnnotatedParameterProcessor所有默认的PathVariableParameterProcessor、RequestParamParameterProcessor、RequestHeaderParameterProcessor、QueryMapParameterProcessor都会失效,为啥会失效我们去看看SpringMvcContract这个类。所以自定义AnnotatedParameterProcessor需要慎重。

结束

由于自己才疏学浅,难免会有纰漏,假如你发现了错误的地方,还望留言给我指出来,我会对其加以修正。

如果你觉得文章还不错,你的转发、分享、赞赏、点赞、留言就是对我最大的鼓励。

感谢您的阅读,十分欢迎并感谢您的关注。

延伸 · 阅读

精彩推荐
  • Java教程20个非常实用的Java程序代码片段

    20个非常实用的Java程序代码片段

    这篇文章主要为大家分享了20个非常实用的Java程序片段,对java开发项目有所帮助,感兴趣的小伙伴们可以参考一下 ...

    lijiao5352020-04-06
  • Java教程xml与Java对象的转换详解

    xml与Java对象的转换详解

    这篇文章主要介绍了xml与Java对象的转换详解的相关资料,需要的朋友可以参考下...

    Java教程网2942020-09-17
  • Java教程Java BufferWriter写文件写不进去或缺失数据的解决

    Java BufferWriter写文件写不进去或缺失数据的解决

    这篇文章主要介绍了Java BufferWriter写文件写不进去或缺失数据的解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望...

    spcoder14552021-10-18
  • Java教程Java使用SAX解析xml的示例

    Java使用SAX解析xml的示例

    这篇文章主要介绍了Java使用SAX解析xml的示例,帮助大家更好的理解和学习使用Java,感兴趣的朋友可以了解下...

    大行者10067412021-08-30
  • Java教程Java实现抢红包功能

    Java实现抢红包功能

    这篇文章主要为大家详细介绍了Java实现抢红包功能,采用多线程模拟多人同时抢红包,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙...

    littleschemer13532021-05-16
  • Java教程小米推送Java代码

    小米推送Java代码

    今天小编就为大家分享一篇关于小米推送Java代码,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧...

    富贵稳中求8032021-07-12
  • Java教程Java8中Stream使用的一个注意事项

    Java8中Stream使用的一个注意事项

    最近在工作中发现了对于集合操作转换的神器,java8新特性 stream,但在使用中遇到了一个非常重要的注意点,所以这篇文章主要给大家介绍了关于Java8中S...

    阿杜7472021-02-04
  • Java教程升级IDEA后Lombok不能使用的解决方法

    升级IDEA后Lombok不能使用的解决方法

    最近看到提示IDEA提示升级,寻思已经有好久没有升过级了。升级完毕重启之后,突然发现好多错误,本文就来介绍一下如何解决,感兴趣的可以了解一下...

    程序猿DD9332021-10-08