分享一个 SpringCloud Feign 中所埋藏的坑
背景
前段时间同事碰到一个问题,需要在 SpringCloud
的 Feign 调用中使用自定义的 URL;通常情况下是没有这个需求的;毕竟都用了 SpringCloud
的了,那服务之间的调用都是走注册中心的,不会需要自定义 URL 的情况。
但也有特殊的,比如我们这里碰到 ToB
场景,需要对每个商户自定义的 URL
进行调用。
虽说也可以使用原生的 Feign
甚至是自定义一个 OKHTTP Client
实现,但这些方案都得换一种写法;
打算利用现有的 SpringCloud
OpenFeign
来实现,毕竟原生的 Feign 其实是支持该功能的,而 SpringCloud OpenFeign
也只是在这基础上封装了一层。
只需要在接口声明处加上一个 URI
参数即可,这样就可以在每次调用时传递不同的 URI
来实现动态 URL
的目的。
想法很简单,但实践起来却不是那么回事了。
伪代码如下:
@FeignClient(name = "dynamic")
interface DynamicClient {
@GetMapping("/")
String get(URI uri);
}
dynamicClient.get(URI.create("https://github.com"));
执行后会抛出负载均衡的异常:
java.lang.RuntimeException: com.netflix.client.ClientException:
Load balancer does not have available server for client: github.com
这个异常也能理解,就是找不到 github 这个服务;找不到也是合理的,毕竟也不是一个内部注册的服务。
但按照 Feign
的官方介绍,只要接口中声明了 URI
这个参数就能自定义,同时我自己也用原生的 Feign 测试过确实没什么问题。
Debug
那问题只能出在 SpringCloud OpenFeign
的封装上了;经过同事的搜索在网上找到一篇博客解决了这个问题。
https://www.cnblogs.com/syui-terra/p/14386188.html
按照文中的说法,确实只需要加上 URL 参数同时有值就可以了,但原因不明。
本着打破砂锅问到底的精神,我个人也想知道 OpenFeign
是如何处理的,只要 url 有值就可以,这完全是个黑盒,而且在官方的注释中并没有对这种情况有特殊说明。
所以我准备从源码中找到答案。
既然是 url 有值就能正常运行,那一定是在运行过程中获取了这个值;
但我在源码中查看 url 所使用的地方,并没有在单测之外找到哪里有所应用,说明源码中并没有直接调用 url()
这个函数来获取值。
但 org.springframework.cloud.openfeign.FeignClient
这个注解总会使用吧,于是我又查询这个注解的使用情况。
最终在这里查到了使用的痕迹。
这里查阅源码时也有一些小技巧,比如如果我们直接查询时,IDEA 默认的查询范围是整个项目和所有依赖库,会有许多干扰信息。
比如我这里就需要只看项目源码,单测这些都不用看;所以在查询的时候可以过滤一下,这样干扰信息就会少很多。
左边的工具栏还有许多过滤条件,大家可以自行研究一下。
接着从源码中进行阅读,会发现是将 @FeignClient
中的所有数据都写到一个 Map
里进行使用的。
最终会发现这个 url 被写入到了 FeignClientFactoryBean
中的 url 成员变量中了。
查看哪里在使用这个 url 就知道背后的原理了。
在这里打个断点会发现:当 url 为空时会返回一个 LoadBalance
的 client
,也就是会从注册中心获取 url
的客户端,而 url
有值时则会获取一个默认的客户端,这样就不会走负载均衡了。
所以我们如果想在 OpenFeign 中使用动态 url 时就得让 @Feign 的 url 有值才行,无论是什么都可以。
Feign 的实现
既然已经看到这一步了,我也比较好奇 Feign 是如何做到只要有 URI 参数就使用指定的 URL 呢?
这里也分享一个读源码的小技巧,如果我们跟着程序执行的思路去一步步
debug
的话会非常消耗时间,毕竟这类成熟库的代码量也不小。
这里我们从官方文档中可以得知只要在接口参数中使用了 java.net.URI
便会走自定义的 url,所以我们反过来只要在源码中找到哪里在使用 java.net.URI
便能知道关键源码。
毕竟使用 java.net.URI
的场景也不会太多。
所以只需要在这个依赖的地方 cmd+shift+f
全局搜索 java.net.URI
就能查到结果,果然不多,只有两处使用。
再结合使用场景猜测大概率是判断参数中是否是有 URL.class
这样的条件,或者是 url 对象;总之我们先用
URL
这样关键字在这两个文件中搜索一下,记得勾选匹配大小写;最后会发现的确是判断了参数中是否有 URL
这个类,同时将这个索引位置记录了下来。
想必后续会通过这个索引位置读取最终的 url
信息。
最终通过这个索引的使用地方查询到了核心源码,如果有值时就取这个 URI 中所指定的地址作为 target
。
到此为止这个问题的背后原理都已经分析完毕了。
总结
其实本文重点是分析了一些 debug
和阅读源码的一些小技巧,特别是在读关于 Spring
相关的代码时一定不能 debug 跟踪到细节中,因为调用链通常是很长的,稍不留神就把自己都绕晕了,只需要知道核心、关键源码是如何处理的即可。
最后对于 OpenFeign 处理动态 url 的方案确实也有些疑惑,是一个典型的约定大于配置
的场景,但问题就在于我们并不知道这个约定是 @Feign
的 url 得有值。
所以我也提了一个 PR
给 OpenFeign
,感兴趣的朋友也可以查看一下:
https://github.com/spring-cloud/spring-cloud-openfeign/pull/713
分享一个 SpringCloud Feign 中所埋藏的坑的更多相关文章
- 分享一个移动项目中消除click事件点击延迟的方法
对于前端工程师来说,apicloud无疑给我们提供了很好的平台,有各种各样的模块供我们使用,但是在实际项目的时候,很大部分的代码,还是需要我们用html css js来实现的.但是呢,移动端页面对于c ...
- 分享一个在js中判断数据是undefined,NaN,null,的技巧
教大家如何在js中判断一个值是否是undefined,null,NaN,以及如何单独判断 平常开发过程中大家可能遇到一种问题,就是取页面某个值的时候获取不到这个var就是undefined了,如果是数 ...
- 分享一个vueui axios-mock-adapter 中的用法
import axios from 'axios'; import MockAdapter from 'axios-mock-adapter'; import { LoginUsers, Users ...
- 分享一个linux系统中采用嵌套for循环比较两个数组内容,并输出相同值的shell脚本
#!/bin/bash array1=(1 3 5 6 7 9) array2=(3 4 9) echo array1=${array1[@]} echo array2=${array2[@]} fo ...
- 分享一个linux系统中循环遍历两个数组内容,并输出数组中的不同内容的shell脚本
cat diffarray.sh #!/bin/bash arry_list1=(1 2 3 4 5 6 7 8 9) arry_list2=(3 5 8) declare -a diff_list ...
- SpringCloud Feign context-path踩到的坑
最近在使用SpringCloud的context-path时,遇到了一些坑,记录一下. server.context-path(上下文) 服务提供者的application配置文件中有一个属性叫ser ...
- 分享一个SQLSERVER脚本(计算数据库中各个表的数据量和每行记录所占用空间)
分享一个SQLSERVER脚本(计算数据库中各个表的数据量和每行记录所占用空间) 很多时候我们都需要计算数据库中各个表的数据量和每行记录所占用空间 这里共享一个脚本 CREATE TABLE #tab ...
- 分享一个安卓中异步获取网络图片并自适应大小的第三方程序(来自github)
安卓中获取网络图片,生成缓存 用安卓手机,因为手机流量的限制,所以我们在做应用时,要尽量为用户考虑,尽量少耗点用户的流量,而在应用中网络图片的显示无疑是消耗流量最大的,所以我们可以采取压缩图片或者将图 ...
- (转)分享一个SQLSERVER脚本(计算数据库中各个表的数据量和每行记录所占用空间)
分享一个SQLSERVER脚本(计算数据库中各个表的数据量和每行记录所占用空间) 很多时候我们都需要计算数据库中各个表的数据量和每行记录所占用空间 这里共享一个脚本 CREATE TABLE #tab ...
随机推荐
- 【网易云信】H5 容器技术方案
Native 开发原生应用是手机操作系统厂商(目前主要是苹果的 iOS 和 Google 的 Android)对外界提供的标准化的开发模式,他们对于 Native 开发提供了一套标准化实现和优化方案. ...
- 如何利用WebSocket实现网页版聊天室
花了将近一周的时间终于完成了利用WebSocket完成网页版聊天室这个小demo,期间还走过了一段"看似弯曲"的道路,但是我想其实也不算是弯路吧,因为你走过的路必将留下你的足迹.这 ...
- Shiro 安全框架详解一(概念+登录案例实现)
shiro 安全框架详细教程 总结内容 一.RBAC 的概念 二.两种常用的权限管理框架 1. Apache Shiro 2. Spring Security 3. Shiro 和 Spring Se ...
- Photoshop之用“色彩范围”命令抠像
1. 打开一个文件.执行"选择>色彩范围",勾选"本地化颜色族",然后在任务背景上单击取样. 2. 取好样以后点击确定,图片如下所示,执行"选择 ...
- wx.getImageInfo和wx.downloadFile下载用户头像报错(小程序canvas以及小程序图片下载部分)
我先上图 之前我们后台配置的 downloadFile 合法域名是 https://wx.qlogo.cn, 用了好久都没出问题, 前段时间, 用户反馈 分享海报, 用户头像出不来!!!! ...
- kafka快速入门到精通
目录 1. 消息队列两种模式 1.1 消息队列作用 1.2 点对点模式(一对一,消费者主动拉取数据,消息收到后消息删除) 1.3 发布/订阅模式(一对多,消费数据之后不会删除消息) 1.4 kafka ...
- power app 解决方案中表导入问题
我们在powerapp中导出的表,解压后是会是乱码,导致在导入的时候会失败,或者导入数据不全. 使用 2 但是直接导入也会是乱码,所以需要将文件重新保存一下: 首先新建一个excel 选择要导入的那个 ...
- Python-初见
目录 概述 关键字 标准数据类型 Number String List Tuple Set Dictionary 删除对象 数据类型转换 推导式 运算符 迭代器与生成器 迭代器 生成器 函数 参数传递 ...
- redis5.0.0集群搭建【实战经历】
redis集群搭建 作者:陈土锋 时间:2020年6月2日 目录 一.环境介绍... 1 1.机器准备... 1 2.关闭防护墙和selinux. 1 3.时间同步... 1 二.Redis Clus ...
- Java 8 学习记录
Java 8 学习记录 官方文档 https://docs.oracle.com/javase/8/ https://docs.oracle.com/javase/8/docs/index.html ...