乐趣区

关于springcloud:SpringCloud-Alibaba微服务实战二十三-Feign-性能调优

概述

在失常状况下 Feign 有三种客户端实现:

  1. Client.Default类:默认的 feign.Client 客户端实现类,外部应用HttpURLConnnection 实现 HTTP URL 申请解决;
  2. ApacheHttpClient 类:外部应用 Apache httpclient 开源组件实现 HTTP URL 申请解决的 feign.Client 客户端实现类;
  3. OkHttpClient类:外部应用OkHttp3 开源组件实现 HTTP URL 申请解决的 feign.Client 客户端实现类。
`@ConditionalOnClass({ILoadBalancer.class, Feign.class})`
`@ConditionalOnProperty(value = "spring.cloud.loadbalancer.ribbon.enabled",`
 `matchIfMissing = true)`
`@Configuration(proxyBeanMethods = false)`
`@AutoConfigureBefore(FeignAutoConfiguration.class)`
`@EnableConfigurationProperties({FeignHttpClientProperties.class})`
`@Import({ HttpClientFeignLoadBalancedConfiguration.class,`
 `OkHttpFeignLoadBalancedConfiguration.class,`
 `DefaultFeignLoadBalancedConfiguration.class })`
`public class FeignRibbonClientAutoConfiguration {`
 `...`
`}`

在后面一节内容中咱们看到 Feign 默认客户端实现 HttpURLConnnection性能不是很好,与 Dubbo RPC 的性能相差很大。基于 HttpURLConnnection的测试后果如下:

本章内容咱们须要对 Feign 的所有客户端进行性能测试,以此来确定抉择一个最优的客户端调用工具。

测试工具

测试服务器:Intel Core i5-7200U CPU @ 2.50GHz 2.70GHz  6 核 16G 内存

测试工具:JMeter5.1

线程数:1000

Ramp-Up : 10

JMeter 测试工具

「ps : 本文中呈现的所有性能测试后果我都至多测试过 10 遍以上,最初选取的是一个绝对均匀的后果,测试后果绝对还是比拟精确的。」

HttpClient

首先咱们先将客户端工具切换到 HttpClient,查看 HttpClientFeignLoadBalancedConfiguration 配置类,源码如下

`@Configuration(proxyBeanMethods = false)`
`@ConditionalOnClass(ApacheHttpClient.class)`
`@ConditionalOnProperty(value = "feign.httpclient.enabled", matchIfMissing = true)`
`@Import(HttpClientFeignConfiguration.class)`
`class HttpClientFeignLoadBalancedConfiguration {`
 `@Bean`
 `@ConditionalOnMissingBean(Client.class)`
 `public Client feignClient(CachingSpringLoadBalancerFactory cachingFactory,`
 `SpringClientFactory clientFactory, HttpClient httpClient) {`
 `ApacheHttpClient delegate = new ApacheHttpClient(httpClient);`
 `return new LoadBalancerFeignClient(delegate, cachingFactory, clientFactory);`
 `}`
`}`

从代码 @ConditionalOnClass({ApacheHttpClient.class})注解可知,只须要在 pom 文件上加上 HttpClient依赖即可。另外须要在配置文件中配置 feign.httpclient.enabledtrue从 @ConditionalOnProperty注解可知,这个配置能够不写,因为在默认状况下就为 true。

所以要应用 HttpClient 咱们只须要在消费者模块 order-service引入 httpclient 依赖即可:

`<dependency>`
 `<groupId>io.github.openfeign</groupId>`
 `<artifactId>feign-httpclient</artifactId>`
`</dependency>`

测试后果

在高并发下性能测试竟然比不过原生的 HttpURLConnnection,有点点悲观!

OkHttp

同样先查看 okhttp 的配置类 OkHttpFeignLoadBalancedConfiguration

`@Configuration(proxyBeanMethods = false)`
`@ConditionalOnClass(OkHttpClient.class)`
`@ConditionalOnProperty("feign.okhttp.enabled")`
`@Import(OkHttpFeignConfiguration.class)`
`class OkHttpFeignLoadBalancedConfiguration {`
 `@Bean`
 `@ConditionalOnMissingBean(Client.class)`
 `public Client feignClient(CachingSpringLoadBalancerFactory cachingFactory,`
 `SpringClientFactory clientFactory, okhttp3.OkHttpClient okHttpClient) {`
 `OkHttpClient delegate = new OkHttpClient(okHttpClient);`
 `return new LoadBalancerFeignClient(delegate, cachingFactory, clientFactory);`
 `}`
`}`

查看注解咱们晓得要应用 OkHttp 必须满足两个条件:

  1. 必须满足OkHttpClient.class 在以后类门路中存在,即引入相应依赖
  2. 必须要配置 feign.okhttp.enabled 配置项的值为 true

所以这里我先引入 OkHttp 的依赖:

`<dependency>`
 `<groupId>io.github.openfeign</groupId>`
 `<artifactId>feign-okhttp</artifactId>`
`</dependency>`

而后批改配置文件,开启 OkHttp:

`feign:`
`...`
 `okhttp:`
 `enabled: true`

测试后果

三个客户端中性能最好的!

测试后果剖析

通过下面测试后果(默认配置)咱们很显著能够得出以下两个论断:

  1. OKHttp 性能优于其余两种,举荐应用!
  2. 在高并发状况下 Apache httpclient 效率甚至还没有默认的 HttpURLConnection 效率高。且在压测过程中,发现应用连接池申请卡顿景象很容易呈现,apache httpclient 甚至还呈现申请卡死状况。httpclient 最不举荐应用。

容器优化

Undertow是一个用 java 编写的灵便的高性能 Web 服务器,提供基于 NIO 的阻塞和非阻塞 API。相比于 tomcatUndertow的性能更高,更轻量。借此机会咱们刚好看看 UndertowOkHttp的测试成果。

首先咱们须要引入 undertow 的依赖并排除 tomcat 的依赖:

`<dependency>`
 `<groupId>org.springframework.boot</groupId>`
 `<artifactId>spring-boot-starter-web</artifactId>`
 `<exclusions>`
 `<exclusion>`
 `<groupId>org.springframework.boot</groupId>`
 `<artifactId>spring-boot-starter-tomcat</artifactId>`
 `</exclusion>`
 `</exclusions>`
`</dependency>`
`<dependency>`
 `<groupId>org.springframework.boot</groupId>`
 `<artifactId>spring-boot-starter-undertow</artifactId>`
`</dependency>`

测试成果

通过测试成果能够看出,应用了 undertow 容器后,性能又有了小小的晋升。

总结

本文中的所有测试论断都是基于组件的默认配置,对于那些谋求性能而又不想对参数进行调优的能够思考 OkHttp + Undertow的组合,尽管没方法与 dubbo 等 rpc 协定性能相比,然而在 springcloud 架构中的 http 调用,这两者之间的组合性能最高。

最初为了不便大家直观比拟,将 Feign 原生 HttpURLConnnection的测试后果和基于 OkHttp + Undertow的测试后果放一起供大家参考:

  • HttpURLConnnection

  • OkHttp + Undertow


干货分享

这里为大家筹备了一份小小的礼物,关注公众号,输出如下代码,即可取得百度网盘地址,无套路支付!

001:《程序员必读书籍》
002:《从无到有搭建中小型互联网公司后盾服务架构与运维架构》
003:《互联网企业高并发解决方案》
004:《互联网架构教学视频》
006:《SpringBoot 实现点餐零碎》
007:《SpringSecurity 实战视频》
008:《Hadoop 实战教学视频》
009:《腾讯 2019Techo 开发者大会 PPT》

010:微信交换群

退出移动版