OkHttpClient的最佳实践优化方案

废话不多说,直接上代码

java 复制代码
public class OkHttpExample {


    // 同步 GET 请求
    private static void syncGet(String url) {
        OkHttpClient client = new OkHttpClient.Builder()
                .connectTimeout(3, TimeUnit.SECONDS)
                .readTimeout(3, TimeUnit.SECONDS)
                .build();
        Request request = new Request.Builder()
                .url(url)
                .addHeader("Accept", "application/json")
                .build();

        try (Response response = client.newCall(request).execute()) {
            if (response.isSuccessful()) {
                System.out.println("GET 成功: " + response.body().string());
            } else {
                System.err.println("GET 失败: HTTP " + response.code());
            }
        } catch (IOException e) {
            System.err.println("网络异常: " + e.getMessage());
        }
    }

   
}

这样存在一个问题,不停地实例化OkHttpClient,这样会浪费资源,且在高并发下,会存在性能隐患。所以,以上代码可以优化为下面这样子

java 复制代码
public class OkHttpExample {

    private static final OkHttpClient client = new OkHttpClient.Builder()
            .connectTimeout(3, TimeUnit.SECONDS)
            .readTimeout(3, TimeUnit.SECONDS)
            .build();

    // GET 请求
    private static void syncGet(String url) {
        Request request = new Request.Builder()
                .url(url)
                .addHeader("Accept", "application/json")
                .build();

        try (Response response = client.newCall(request).execute()) {
            if (response.isSuccessful()) {
                System.out.println("GET 成功: " + response.body().string());
            } else {
                System.err.println("GET 失败: HTTP " + response.code());
            }
        } catch (IOException e) {
            System.err.println("网络异常: " + e.getMessage());
        }
    }

   
}

这样是不是没有优化空间了呢?看源码后,其实还有进一步代码空间,就okhttp3.ConnectionPool,这个就是连接池,这也就是不建议重复实例化OkHttpClient原因之一。

java 复制代码
public ConnectionPool() {
    this(5, 5, TimeUnit.MINUTES);
}

默认是设置5,对于绝大部分场景已经够用了,但如果还想进一步优化,可以修改成这样

java 复制代码
public class OkHttpExample {

    private static final OkHttpClient client = new OkHttpClient.Builder()
            .connectionPool(new ConnectionPool(20, 60, TimeUnit.SECONDS))
            .connectTimeout(3, TimeUnit.SECONDS)
            .readTimeout(3, TimeUnit.SECONDS)
            .build();

    // 同步 GET 请求
    private static void syncGet(String url) {
        Request request = new Request.Builder()
                .url(url)
                .addHeader("Accept", "application/json")
                .build();

        try (Response response = client.newCall(request).execute()) {
            if (response.isSuccessful()) {
                System.out.println("GET 成功: " + response.body().string());
            } else {
                System.err.println("GET 失败: HTTP " + response.code());
            }
        } catch (IOException e) {
            System.err.println("网络异常: " + e.getMessage());
        }
    }

   
}

若调用方是消息消费者/定时任务,推荐异步 enqueue(),但需确保回调中不阻塞,这种在安卓开发常使用。这个是通过okhttp3.Dispatcher实现,这个也可以进行调整。

java 复制代码
 OkHttpClient client = new OkHttpClient.Builder()
    .dispatcher(new Dispatcher().apply(d -> {
        d.setMaxRequests(128); // 全局最大并发(默认64)
        d.setMaxRequestsPerHost(12);// 单域名最大并发(默认5)
    }))
    .build();   
java 复制代码
    // 异步 GET 请求(推荐用于 Android/UI 线程)
    private static void asyncGet(String url) {
        Request request = new Request.Builder()
                .url(url)
                .build();

        client.newCall(request).enqueue(new Callback() {
            @Override
            public void onFailure(Call call, IOException e) {
                System.err.println("异步请求失败: " + e.getMessage());
            }

            @Override
            public void onResponse(Call call, Response response) throws IOException {
                // 注意:回调线程非主线程,Android 中更新 UI 需切回主线程
                try (Response res = response) { // 手动关闭,避免泄漏
                    if (res.isSuccessful()) {
                        System.out.println("异步 GET 成功: " + res.body().string());
                    } else {
                        System.err.println("异步响应异常: " + res.code());
                    }
                }
            }
        });
    }
相关推荐
Eric 辰东1 分钟前
【C 语言程序的编译和链接】详解编译链接过程
c语言·笔记·算法·学习方法
星恒随风1 分钟前
C语言数据结构排序算法详解(上):从插入排序、希尔排序到选择排序、堆排序
c语言·数据结构·笔记·学习·排序算法
方也_arkling3 分钟前
【Java-Day09】继承
java·开发语言
西安邮电大学3 分钟前
Kafka保证消息顺序性
java·后端·kafka
迈巴赫车主4 分钟前
蓝桥杯21247弹跳鞋java
java·开发语言·数据结构·算法·职场和发展·蓝桥杯
xinhuanjieyi5 分钟前
JavaFX WebView 不支持 Brotli (br) 压缩编码警告修复
java
Adair_z10 分钟前
[SEO艺术重读] 第13篇 SEO教育与研究
java·网络·数据库
疯狂打码的少年14 分钟前
指令寻址方式(立即、直接、间接、变址等)
网络·笔记
utf8mb4安全女神33 分钟前
⽇志管理与深层防⽕墙
java·开发语言·spring boot
better_liang33 分钟前
每日Java面试场景题知识点之-数据库与缓存的一致性
java·数据库·redis·面试·分布式系统·缓存一致性·cache aside