httpClient同步、异步性能对比
创始人
2024-04-03 17:23:30
0

0、测试目的

同步阻塞模式下,如果服务端接口响应较慢,那会直接影响客户端接口请求的吞吐量,虽然可以通过在应用代码中通过异步线程的方式优化,但是会增加客户端的线程开销。所以考虑用异步模式来解决这个问题

因此测试时,主要是针对线程数设置比较小的情况下,客户端发起请求的吞吐量来进行对比

1、准备工作

用spring boot写一个最简单的接口:sleep 1s,然后返回ok
在这里插入图片描述

客户端程序引入httpClient依赖:

org.apache.httpcomponents.client5httpclient55.1.3

2、同步模式

代码:

import lombok.SneakyThrows;
import org.apache.hc.client5.http.ClientProtocolException;
import org.apache.hc.client5.http.classic.methods.HttpGet;
import org.apache.hc.client5.http.impl.classic.CloseableHttpClient;
import org.apache.hc.client5.http.impl.classic.HttpClients;
import org.apache.hc.client5.http.impl.io.PoolingHttpClientConnectionManager;
import org.apache.hc.core5.http.ClassicHttpResponse;
import org.apache.hc.core5.http.HttpEntity;
import org.apache.hc.core5.http.HttpStatus;
import org.apache.hc.core5.http.ParseException;
import org.apache.hc.core5.http.io.HttpClientResponseHandler;
import org.apache.hc.core5.http.io.entity.EntityUtils;import java.io.IOException;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger;public class SyncClientHttpTest {static final CloseableHttpClient httpclient;static {PoolingHttpClientConnectionManager connectionManager = new PoolingHttpClientConnectionManager();connectionManager.setMaxTotal(1000);connectionManager.setDefaultMaxPerRoute(100);httpclient = HttpClients.custom().setConnectionManager(connectionManager).build();}public static void main(final String[] args) throws Exception {AtomicInteger atomicInteger = new AtomicInteger(0);AtomicBoolean stop = new AtomicBoolean(false);for (int i = 0; i < 10; i++) {new Thread(() -> {while (!stop.get()) {httpGet();atomicInteger.incrementAndGet();}}).start();}Thread.sleep(30000);stop.set(true);Thread.sleep(1000);System.out.println(atomicInteger.get());System.exit(0);}@SneakyThrowsprivate static void httpGet() {final HttpGet httpget = new HttpGet("http://localhost:8080/test");// Create a custom response handlerfinal HttpClientResponseHandler responseHandler = new HttpClientResponseHandler() {@Overridepublic String handleResponse(final ClassicHttpResponse response) throws IOException {final int status = response.getCode();if (status >= HttpStatus.SC_SUCCESS && status < HttpStatus.SC_REDIRECTION) {final HttpEntity entity = response.getEntity();try {return entity != null ? EntityUtils.toString(entity) : null;} catch (final ParseException ex) {throw new ClientProtocolException(ex);}} else {throw new ClientProtocolException("Unexpected response status: " + status);}}};final String responseBody = httpclient.execute(httpget, responseHandler);
//            System.out.println(responseBody);if(!responseBody.equals("ok")) {throw new RuntimeException("error");}}
}}

开启5个线程,循环发起请求30s

打印结果:150
差不多每秒5个请求,符合预期

改为10个线程
打印结果:300

改为100个线程
打印结果:3000

请求吞吐和线程数呈线性增长关系(线程数应小于maxPerRoute)

3、异步模式

代码:

import lombok.SneakyThrows;
import org.apache.hc.client5.http.async.methods.*;
import org.apache.hc.client5.http.impl.async.CloseableHttpAsyncClient;
import org.apache.hc.client5.http.impl.async.HttpAsyncClients;
import org.apache.hc.client5.http.impl.nio.PoolingAsyncClientConnectionManager;
import org.apache.hc.core5.concurrent.FutureCallback;
import org.apache.hc.core5.reactor.IOReactorConfig;
import org.apache.hc.core5.util.Timeout;import java.util.concurrent.Future;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger;/*** Example of asynchronous HTTP/1.1 request execution.*/
public class AsyncClientHttpTest {static final CloseableHttpAsyncClient client;static final AtomicInteger atomicInteger = new AtomicInteger(0);static final AtomicBoolean stop = new AtomicBoolean(false);static {PoolingAsyncClientConnectionManager connectionManager = new PoolingAsyncClientConnectionManager();connectionManager.setMaxTotal(1000);connectionManager.setDefaultMaxPerRoute(100);IOReactorConfig ioReactorConfig = IOReactorConfig.custom().setSoTimeout(Timeout.ofSeconds(5)).setIoThreadCount(5) //IO线程数.build();client = HttpAsyncClients.custom().setIOReactorConfig(ioReactorConfig).setConnectionManager(connectionManager).build();client.start();}public static void main(final String[] args) throws Exception {new Thread(()->{while (!stop.get()) {httpGet();}}).start();Thread.sleep(5000);stop.set(true);Thread.sleep(25000);System.out.println(atomicInteger.get());//        client.close(CloseMode.GRACEFUL);System.exit(0);}@SneakyThrowsprivate static void httpGet() {final SimpleHttpRequest request = SimpleRequestBuilder.get().setUri("http://localhost:8080//test").build();final Future future = client.execute(SimpleRequestProducer.create(request),SimpleResponseConsumer.create(),new FutureCallback() {@Overridepublic void completed(final SimpleHttpResponse response) {
//                        System.out.println(request + "->" + new StatusLine(response));
//                        System.out.println(response.getBody().getBodyText());if(!response.getBody().getBodyText().equals("ok")) {throw new RuntimeException("error");}atomicInteger.incrementAndGet();}@Overridepublic void failed(final Exception ex) {System.out.println(request + "->" + ex);}@Overridepublic void cancelled() {System.out.println(request + " cancelled");}});}}

ps: 这里代码其实不够严谨,不过测试结果对比已经很悬殊了,不影响最终结论

开启5个IO线程(不设置默认为cpu核数)
客户端1个线程循环发起请求5s,之后再sleep 25s打印结果

打印结果:2700

修改代码:connectionManager.setDefaultMaxPerRoute(100);
->connectionManager.setDefaultMaxPerRoute(200);
调大maxPerRoute为200

打印结果:5400

可以看到异步模式下,每秒的吞吐受maxPerRoute的影响较大(基本持平)
注意如果不手动设置,这个默认值为5,所以如果不进行ConnectionManager设置,异步的测试结果会很差

3、结论

异步模式下因为使用了多路复用,一个IO线程管理多个连接,所以只需少量线程即可进行大量的远程接口调用

相关内容

热门资讯

保存时出现了1个错误,导致这篇... 当保存文章时出现错误时,可以通过以下步骤解决问题:查看错误信息:查看错误提示信息可以帮助我们了解具体...
汇川伺服电机位置控制模式参数配... 1. 基本控制参数设置 1)设置位置控制模式   2)绝对值位置线性模...
不能访问光猫的的管理页面 光猫是现代家庭宽带网络的重要组成部分,它可以提供高速稳定的网络连接。但是,有时候我们会遇到不能访问光...
表格中数据未显示 当表格中的数据未显示时,可能是由于以下几个原因导致的:HTML代码问题:检查表格的HTML代码是否正...
本地主机上的图像未显示 问题描述:在本地主机上显示图像时,图像未能正常显示。解决方法:以下是一些可能的解决方法,具体取决于问...
表格列调整大小出现问题 问题描述:表格列调整大小出现问题,无法正常调整列宽。解决方法:检查表格的布局方式是否正确。确保表格使...
不一致的条件格式 要解决不一致的条件格式问题,可以按照以下步骤进行:确定条件格式的规则:首先,需要明确条件格式的规则是...
Android|无法访问或保存... 这个问题可能是由于权限设置不正确导致的。您需要在应用程序清单文件中添加以下代码来请求适当的权限:此外...
【NI Multisim 14...   目录 序言 一、工具栏 🍊1.“标准”工具栏 🍊 2.视图工具...
银河麒麟V10SP1高级服务器... 银河麒麟高级服务器操作系统简介: 银河麒麟高级服务器操作系统V10是针对企业级关键业务...