# Spring Boot 集成 OkHttp

作者:Tom哥
公众号:微观技术
博客:https://offercome.cn (opens new window)
人生理念:知道的越多,不知道的越多,努力去学

Spring Boot 作为主流微服务框架,拥有成熟的社区生态。市场应用广泛,为了方便大家,整理了一个基于spring boot的常用中间件快速集成入门系列手册,涉及RPC、缓存、消息队列、分库分表、注册中心、分布式配置等常用开源组件,大概有几十篇文章,陆续会开放出来,感兴趣同学可以关注&收藏

# 简介

OKHttp 是 Square 公司开发的一款网络框架,其设计和实现的目的就是高效。OkHttp框架完整的实现了 HTTP 协议,支持的协议有 HTTP/1.1、SPDY、HTTP/2.0 。在 Android 4.4 的源码中HttpURLConnection 已经被替换成OkHttp。

既然 OkHttp 的招牌是高效,那它是如何实现的呢?

  • 采用连接池技术减少
  • 默认使用 GZIP 数据压缩格式,降低传输内容的大小
  • 采用缓存避免重复的网络请求
  • 支持SPDY、HTTP/2.0,对于同一主机的请求可共享同一socket连接
  • 若SPDY或HTTP/2.0 不可用,还会采用连接池提高连接效率
  • 网络出现问题、会自动重连(尝试连接同一主机的多个ip地址)
  • 使用 okio 库简化数据的访问和存储

# 常用概念

# OkHttpClient

OkHttpClient 可以说是整个OkHttp 框架的一个门面(Facade 模式),在使用 OkHttp 框架时只需同OkHttpClient 打交道,而不用考虑其内部复杂的构成。这个类的职责就是配置参数和生成 Call 对象,配置参数包括:超时时间(连接、读、写)、代理服务器、Dns、协议和连接规范、缓存配置,拦截器、证书验证等。

# Request

Request 是 一个基本的 HTTP 网络请求,包括要请求的URL,同 HTTP 规范一样,它需要包含请求行,请求头,请求体(可以为空)。

# Response

Response 即为网络响应,包括响应行,响应头和响应体(允许为空), 响应体只能被消费一次,且除响应体外的属性不可变。

# Call

Call 是一个接口,实现了对 Request 的封装,可以看做是一次请求任务,可以同步(execute)、异步(enqueue)执行,也可以取消 (cancel) 或者查询当前任务状态 。RealCall 是 Call 的一个子类,对其接口中定义的方法做了实现。

Cache & CacheStrategy

Cache 即缓存,是对于某一特定 Request 的缓存,缓存的内容为该 Request 的 Response ,缓存多以<请求、响应>键值对的形式存在。缓存可以为内存,也可以为硬盘,缓存可以进行失效时间、最大容量、缓存目录等设置。 CacheStrategy 即缓存策略,即当用 OkHttp 请求一个 Request 时,是采用缓存还是网络还是两者皆用。

# 核心流程

  • 构建OkHttpClient对象
  • 构建Request对象
  • 由前两步创建的OkHttpClientRequest对象创建Call对象
  • 通过Call对象发起请求,并得到一个Response对象,解析得到最终返回的结果。

# 代码演示

在pom.xml中添加以下依赖项:

<dependency>
    <groupId>com.squareup.okhttp3</groupId>
    <artifactId>okhttp</artifactId>
    <version>3.10.0</version>
</dependency>
1
2
3
4
5

官方建议是一个工程只配置一个 OKHttpClient 来处理所有的网络请求。

初始化 OkHttpClient 客户端实例

client = new OkHttpClient.Builder()
        .connectTimeout(connTimeout, TimeUnit.MILLISECONDS)
        .readTimeout(readTimeout, TimeUnit.MILLISECONDS)
        .writeTimeout(writeTime, TimeUnit.MILLISECONDS)
        .sslSocketFactory(createSslSocketFactory(), x509TrustManager())
        .connectionPool(createConnectionPool())
        .build();
1
2
3
4
5
6
7

发送post请求

public static String post(String url, String message, Map<String, String> headers) {
    RequestBody requestBody = RequestBody.create(JSON_TYPE, message);
    Builder builder = new Builder();
    builder.url(url).post(requestBody).build();
    Response response = null;
    String result = "";
    try {
        // 添加header
        if (headers != null && !headers.isEmpty()) {
            Iterator iterator = headers.entrySet().iterator();
            while (iterator.hasNext()) {
                Map.Entry entry = (Map.Entry) iterator.next();
                builder.addHeader((String) entry.getKey(), (String) entry.getValue());
            }
        }
        Request request = builder.build();
        response = client.newCall(request).execute();
        if (response.code() == 200) {
            result = response.body() == null ? "" : response.body().string();
        }

    } catch (IOException var3) {
        log.error("[HttpPost JSON Exception] URL:{}, error:{}", url, var3);
        throw new RuntimeException(var3);

    } finally {
        if (response != null) {
            response.close();
        }
    }
    return result;
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32

同步请求使用的是call.execute(),只有当整个请求完成时才会从.execute()方法返回。

而对于异步请求来说,.enqueue(Callback)方法只要调用完就立即返回了,当网络请求返回之后会回调Callback的onResponse/onFailure方法,并且这两个回调方法是在子线程执行的,这也是异步请求和同步请求之间最主要的差别。

本地跑单元测试,发送post模拟请求,请求参数采用json格式数据,content-type = "application/json;charset=utf-8"

@Test
public void queryUser() {
    User user = User.builder().userName("TomGE").age(29).address("北京").build();

    String url = "http://localhost:8090/queryUser";
    String requestJson = JSON.toJSONString(user);
//        System.out.println(requestJson);
    String result = OkHttpUtil.post(url, requestJson, null);
    System.out.println("响应结果:");
    System.out.println(result);
}
1
2
3
4
5
6
7
8
9
10
11

响应结果:

已经接收到请求,用户名:TomGE , 年龄:29 , 地址:北京。响应:sucess!
1

# 项目源码

https://github.com/aalansehaiyang/spring-boot-bulking  

模块:spring-boot-bulking-okhttp
1
2
3
上次更新: 2023/3/4