【Spring Cloud 源码解读】之 【这也太神奇了,RestTemplate加上一个@LoadBalanced注解就能实现负载均衡!】
前提概要:
前天,有个前端大佬问了我两个问题:为啥不引入Ribbon
依赖就能使用Ribbon
?为啥RestTemplate
加上@LoadBalanced
注解就能负载均衡了?我也表示很疑惑,而我自己其实也真的没去了解过,所以趁着工作不太忙,赶紧去研究一波。
第一个问题比较简单,一般都是其他依赖引入了Ribbon
,我这里是Nacos
,而他那边也是注册中心Eureka
。
第二个问题由于有一点深度,所以需要好好的研究一番。
1、准备:启动两个服务提供者实例,然后启动一个服务消费者实例
2、开始搞起来
1、准备两个RestTemplate:
一个启动负载均衡,一个不启动负载均衡
@Configuration
public class MyConfiguration {
// 启动负载均衡
@LoadBalanced
@Bean
RestTemplate loadBalanced() {
return new RestTemplate();
}
// 不启动负载均衡
@Bean
RestTemplate restTemplate() {
return new RestTemplate();
}
}
2、负载均衡探索:
@Autowired
private RestTemplate loadBalanced;
@GetMapping("restTemplate-hello")
public String sayHello(){
return loadBalanced.getForObject("http://cloud-nacos-discovery-server/hello",String.class);
}
注意:使用负载均衡的RestTemplate
去请求时url一定得写服务名,因为Ribbon
会根据服务名[serviceId]
去获取所有实例,然后进行负载均衡。所以记得不能写IP:Port
,不然会报错。
java.lang.IllegalStateException: No instances available for 10.172.29.666
(1)、为何带上@LoadBalanced就能负载均衡?
之所以标记了@LoadBalanced
的RestTemplate
会带有负载均衡的功能,是因为RestTemplate
里面加入LoadBalancerInterceptor
拦截器。我们也可以看到,我们上面的代码使用的loadBalanced
确实有LoadBalancerInterceptor
拦截器。
(2)、拦截器是如何进行负载均衡的?
RestTemplate
的每次请求都会被此拦截,然后利用Ribbon
实现负载均衡逻辑。
public ClientHttpResponse intercept(final HttpRequest request, final byte[] body, final ClientHttpRequestExecution execution) throws IOException {
URI originalUri = request.getURI();
String serviceName = originalUri.getHost();
Assert.state(serviceName != null, "Request URI does not contain a valid hostname: " + originalUri);
//这里是使用负载均衡,而这里的loadBalancer就是Spring Cloud提供的LoadBalancerClient接口的实现类。
return (ClientHttpResponse)this.loadBalancer.execute(serviceName, this.requestFactory.createRequest(request, body, execution));
}
我们也看到,最后是通过(ClientHttpResponse)this.loadBalancer.execute(serviceName, this.requestFactory.createRequest(request, body, execution))
去负载均衡的,而从上图我们也可以看到,我们RestTemplate
的loadBanalcer
是RibbonLoadBalancerClient
,所以说,最后是通过Ribbon
是负载均衡的。
(3)、那究竟是谁帮RestTemplate加上这个拦截器的?而且是什么时候加的?
① LoadBalancerAutoConfiguration配置类
关于@LoadBalanced
自动生效的配置,我们需要来到这个自动配置类:LoadBalancerAutoConfiguration
。
我们可以看到这个配置类上有俩个注解:@ConditionalOnClass({RestTemplate.class})
,@ConditionalOnBean({LoadBalancerClient.class})
,意思是说:只要有RestTemplate
类存在,并且Spring
容器中存在LoadBalancerClient
类型的Bean,这个配置类才会生效。首先,Spring
的web
模块已经提供了RestTemplate
类,而Ribbon
也提供了实现LoadBalancerClient
接口的实现类,所以说上面所有的条件都符合了,该配置类会生效。
@Configuration
@ConditionalOnClass({RestTemplate.class})
@ConditionalOnBean({LoadBalancerClient.class})
@EnableConfigurationProperties({LoadBalancerRetryProperties.class})
public class LoadBalancerAutoConfiguration {
② 一个关键的成员变量
我们可以看到LoadBalancerAutoConfiguration
中有一个成员变量:
//拿到Spring容器内所有的标注有@LoadBalanced注解的RestTemplate们。 注意:是带有@LoadBalanced注解的
@LoadBalanced
@Autowired(required = false)
private List<RestTemplate> restTemplates = Collections.emptyList();
③ RestTemplateCustomizer来加拦截器
我们会先找拦截器相关的代码,因为此时我们都知道负载均衡主要靠的是拦截器,所以,上代码:
@Bean
@ConditionalOnMissingBean
public RestTemplateCustomizer restTemplateCustomizer(final LoadBalancerInterceptor loadBalancerInterceptor) {
return (restTemplate) -> {
// 我们可以看到,如果我们没有自己实现`RestTemplateCustomizer`,就会执行下面的逻辑,而最突兀的就是,它给每一个`RestTemplate`添加了`LoadBalancerInterceptor`,也就是实现负载均衡的重点所在。
List<ClientHttpRequestInterceptor> list = new ArrayList(restTemplate.getInterceptors());
list.add(loadBalancerInterceptor);
restTemplate.setInterceptors(list);
};
}
④ 何时利用RestTemplateCustomizer来给RestTemplate加拦截器
还有另外一段很重要的代码,需要来解读一下:
首先我们得先了解SmartInitializingSingleton
是干嘛的,它的afterSingletonsInstantiated()
方法会在所有的单例Bean初始化完成之后,再去一个一个的去处理。
public interface SmartInitializingSingleton {
void afterSingletonsInstantiated();
}
那么我们就知道了,接下来要解读的代码就是为了处理一个个带有@LoadBalanced
的RestTemplate
们,利用RestTemplateCustomizer
给RestTemplate
们加上拦截器LoadBalancerInterceptor
。
@Bean
public SmartInitializingSingleton loadBalancedRestTemplateInitializerDeprecated(final ObjectProvider<List<RestTemplateCustomizer>> restTemplateCustomizers) {
return () -> {
restTemplateCustomizers.ifAvailable((customizers) -> {
// 遍历上面提及的成员变量,带@LoadBalanced的RestTemplate们
Iterator var2 = this.restTemplates.iterator();
while(var2.hasNext()) {
RestTemplate restTemplate = (RestTemplate)var2.next();
Iterator var4 = customizers.iterator();
while(var4.hasNext()) {
// 利用上面的RestTemplateCustomizer给RestTemplate们加拦截器
RestTemplateCustomizer customizer = (RestTemplateCustomizer)var4.next();
customizer.customize(restTemplate);
}
}
});
};
}
所以最后,我们可以给第三个问题一个答案:在带有@LoadBalanced
注解的RestTemplate
们完成Bean初始化之后,利用RestTemplateCustomizer
给RestTemplate
们加上拦截器LoadBalancerInterceptor
,来实现负载均衡。
3、非负载均衡探索
@Autowrite
private RestTemplate restTemplate;
@GetMapping("restTemplate-hello")
public String sayHello(){
return myRestTemplate.getForObject("http://10.172.29.666:8887/hello",String.class);
}
首先可以看到,RestTemplate不再带有拦截器
而且,我们可以看到,最后接口走的是SimpleBufferingClientHttpRequest
,而不是RibbonLoadBalancerClient
:
到此,关于为什么添加了@LoadBalanced
就能进行负载均衡的分析已经结束。而如果大家对Ribbon
如何进行负载均衡的也很感兴趣,有空再大家一起研究研究
【Spring Cloud 源码解读】之 【这也太神奇了,RestTemplate加上一个@LoadBalanced注解就能实现负载均衡!】的更多相关文章
- 【Spring Cloud 源码解读】之 【如何配置好OpenFeign的各种超时时间!】
关于Feign的超时详解: 在Spring Cloud微服务架构中,大部分公司都是利用Open Feign进行服务间的调用,而比较简单的业务使用默认配置是不会有多大问题的,但是如果是业务比较复杂,服务 ...
- Spring Security4源码解读探寻权限机制
我们知道springSecurity 会在用户登录的时候获取用户的角色权限, 你是一个普通用户可能没有管理员拥有的权限.用户登录后Authentication 获取用户的权限. 不通用户登录系统会生成 ...
- spring core源码解读之ASM4用户手册翻译之一asm简介
第一章:ASM介绍 1.1 ASM动机: 程序的分析,生成,转换技术可以应用到许多场景: 1.程序分析,从简单的语法解析到完整的语义分析,可以应用在程序中找到潜在的bug,发现无用的代码,工程代码的逆 ...
- Spring Cloud源码分析(四)Zuul:核心过滤器
通过之前发布的<Spring Cloud构建微服务架构(五)服务网关>一文,相信大家对于Spring Cloud Zuul已经有了一个基础的认识.通过前文的介绍,我们对于Zuul的第一印象 ...
- Spring:源码解读Spring IOC原理
Spring IOC设计原理解析:本文乃学习整理参考而来 一. 什么是Ioc/DI? 二. Spring IOC体系结构 (1) BeanFactory (2) BeanDefinition 三. I ...
- Spring IoC源码解读——谈谈bean的几种状态
阅读Spring IoC部分源码有一段时间了,经过不断的单步调试和参阅资料,对Spring容器中bean管理有了一定的了解.这里从bean的几个状态的角度出发,研究下IoC容器. 一.原材料 Xml中 ...
- spring beans源码解读之 ioc容器之始祖--DefaultListableBeanFactory
spring Ioc容器的实现,从根源上是beanfactory,但真正可以作为一个可以独立使用的ioc容器还是DefaultListableBeanFactory,因此可以这么说, DefaultL ...
- spring beans源码解读
spring beans下面有如下源文件包: org.springframework.beans, 包含了操作java bean的接口和类.org.springframework.beans.anno ...
- spring beans源码解读之--总结篇
spring beans下面有如下源文件包: org.springframework.beans, 包含了操作java bean的接口和类.org.springframework.beans.anno ...
随机推荐
- 集合 Enumerable Enumerator yield
集合: 通过索引来访问成员,--引申到索引器 的使用 for foreach循环遍历 --引申到 IEnumerable IEnumerator(会引申到yield) 常用的集合操作,add, ins ...
- SELinux 宽容模式(permissive) 强制模式(enforcing) 关闭(disabled) 几种模式之间的转换
http://blog.sina.com.cn/s/blog_5aee9eaf0100y44q.html 在CentOS6.2 中安装intel 的c++和fortran 的编译器时,遇到来一个关于S ...
- sql —— having
在 SQL 中增加 HAVING 子句原因是,WHERE 关键字无法与聚合函数一起使用.HAVING 子句可以让我们筛选分组后的各组数据. 原表: 我们可以对上面数据根据性别这个字段进行分组查询,分别 ...
- PyTorch代码调试利器: 自动print每行代码的Tensor信息
本文介绍一个用于 PyTorch 代码的实用工具 TorchSnooper.作者是TorchSnooper的作者,也是PyTorch开发者之一. GitHub 项目地址: https://github ...
- tensorflow学习笔记(三十四):Saver(保存与加载模型)
Savertensorflow 中的 Saver 对象是用于 参数保存和恢复的.如何使用呢? 这里介绍了一些基本的用法. 官网中给出了这么一个例子: v1 = tf.Variable(..., nam ...
- 从HelloWorld看Knative Serving代码实现
摘要: Knative Serving以Kubernetes和Istio为基础,支持无服务器应用程序和函数的部署并提供服务.我们从部署一个HelloWorld示例入手来分析Knative Servin ...
- 安装visualStudio 出现 cant install Microsoft.TeamFoundation.OfficeIntegration.Resources
本文告诉大家在安装 VisualStudio 时出现cant install Microsoft.TeamFoundation.OfficeIntegration.Resources如何安装 如果在安 ...
- poj 3384 Feng Shui (Half Plane Intersection)
3384 -- Feng Shui 构造半平面交,然后求凸包上最远点对. 这题的题意是给出一个凸多边形区域,要求在其中放置两个半径为r的圆(不能超出凸多边形区域),要求求出两个圆心,使得多边形中没有被 ...
- H3C 数据链路层
- JavaScript中判断整数的方法
一.使用取余运算符判断 任何整数都会被1整除,即余数是0.利用这个规则来判断是否是整数. 1 2 3 4 5 function isInteger(obj) { return obj%1 == ...