OkHttp 缓存实战

2023-05-16

1、简介

在实际业务中可能某些查询数据,不经常变化,为了节省流量、提高响应速度和增强用户体验等,把变化频率小的数据缓存到本地,以实现复用。

OkHttp 的缓存功能使用起来也比较简单和灵活,接下来我们就来看看

2、配置缓存

配置缓存首先需要创建一个Cache 对象,并且指定缓存目录和缓存大小,然后,调用用 OkHttpClient.Builder()cache() 方法来配置创建的缓存对象。如下所示:

    // 缓存大小
    int cacheSize = 10 * 1024 * 1024; // 10 MiB
    // 缓存目录
    File file = new File("F:\\httpcache");
    // 创建缓存对象
    Cache cache = new Cache(file, cacheSize);
    httpClient = new OkHttpClient.Builder()
        // 设置缓存
        .cache(cache)
        .build();

如果在服务端的接口响应中包含了合适 Cache-Control 响应头,那么,OkHttp 就会默认按此响应头,对数据进行缓存。

Cache-Control 响应头是缓存的一个重点,如果包含了此响应头,在网络请求时,会首先判断缓存是否有效,若有效则直接读取缓存数据,如果失效则会重新请求接口数据。

3、拦截器

有些服务端接口,比如老接口或第三方接口,在响应头中不包含Cache-Control,或者缓存已被禁用。这种情况下要想让缓存功能正常工作,就需要使用自定义拦截器,通过拦截器在给请求的响应中添加合适的Cache-Control响应头即可。如下所示:

// 自定义缓存拦截器
public class CacheInterceptor implements Interceptor{
    private static final String CACHE_CONTROL = "Cache-Control";
    // 缓存时间
    private static final int MAX_AGE = 60;
    private static final String STR_MAX_AGE = "max-age=" + MAX_AGE;

    @NotNull
    @Override
    public Response intercept(@NotNull Chain chain) throws IOException {
        Response response = chain.proceed(chain.request());
        return  response.newBuilder()
            .removeHeader("pragma")
            .addHeader(CACHE_CONTROL, STR_MAX_AGE)
            .build();
    }
}


// 缓存大小
int cacheSize = 10 * 1024 * 1024; // 10 MiB
// 缓存目录
File file = new File("F:\\httpcache");
// 创建缓存对象
Cache cache = new Cache(file, cacheSize);
httpClient = new OkHttpClient.Builder()
    // 设置缓存
    .cache(cache)
    // 添加缓存拦截器
    .addNetworkInterceptor(new CacheInterceptor())
    .build();

4、缓存控制器

以上情况,无论是服务端响应包含Cache-Control 头信息,还是通过拦截器设置的此头信息都属于全局配置,即所有的请求都会缓存,且缓存的时间相同。在实际业务中,可能是有些接口不需要缓存,或者不同接口要求缓存的时间要求不同。要解决这个问题有如下两种办法:

  • 在拦截器中根据 请求路径 request.url() 判断设置数据缓存时间。此种方式不优雅,在这就不考虑了
  • 使用 OkHttp提供的缓存控制器CacheControl来处理。

OkHttp提供了如下两种默认的缓存控制器:

  • CacheControl.FORCE_CACHE 强制使用本地缓存,若缓存不存在则返回一个 code 为 504 的响应
  • CacheControl.FORCE_NETWORK 强制使用网络请求

除了上面提供的默认缓存控制器外,还可以通过 CacheControl.Builder() 构建自定义的缓存控制器,可选的设置方法如下:

  • noCache() 不使用缓存,使用网络请求
  • noStore 不使用缓存,也不存储缓存数据
  • maxAge() 缓存的有效时间,超过此时间会重新请求数据
  • maxStale() 超过缓存有效时间后,可继续使用旧缓存的时间,之后需要重新请求数据。
  • minFresh() 增加额外的缓存有效时间,之后需要重新请求数据
  • onlyCached 只是用缓存,不使用网络请求
  • noTransform() 不接受经过转码的响应
  • immutable() 缓存有效时间内,响应不会变化,避免服务端处理 304 响应

构建一个自定义缓存器如下所示:

// 构建自定义 缓存控制器
CacheControl cacheControl = new CacheControl.Builder()
    .maxAge(10, TimeUnit.SECONDS)
    .maxStale(10, TimeUnit.SECONDS)
    .build();

Request request = new Request.Builder()
    .get()
    // 设置自定义缓存控制器
    .cacheControl(cacheControl)
    .url("http://localhost:10010/index?name=zhangsan&age=20")
    .build();

Call call = httpClient.newCall(request);
Response response = call.execute();

当通过 CacheControl 类设置的缓存时间大于 Cache-Control 响应头时间时,缓存的有效时间为Cache-Control响应头时间,否则使用CacheControl 类设置的时间。

基于此,所以我们可以给有需要的接口请求通过CacheControl类设置缓存策略,然后在拦截器中判断请求是否包含Cache-Control请求头,如果有就把Cache-Control请求头添加到响应中去,这样问题就解决了,修改后的拦截器如下:

 // 自定义缓存拦截器
    public class CacheInterceptor implements Interceptor{
        private static final String CACHE_CONTROL = "Cache-Control";
        // 缓存时间
        private static final int MAX_AGE = 60;
        private static final String STR_MAX_AGE = "max-age=" + MAX_AGE;

        @NotNull
        @Override
        public Response intercept(@NotNull Chain chain) throws IOException {
            Request request = chain.request();
            Response response = chain.proceed(request);
            String header = request.header(CACHE_CONTROL);
            if(header != null && !"".equals(header.trim())){
                return  response.newBuilder()
                        .removeHeader("pragma")
                        .addHeader(CACHE_CONTROL, header)
                        .build();
            }else{
                return response;
            }
        }
    }

5、总结

OkHttp中也可以使用缓存来减少网络请求。在OkHttp可以通过响应头中的Cache-Control 控制缓存的有效时间,在服务端无法提供Cache-Control 响应头时,可以通过自定义拦截器,在拦截器中对请求响应添加Cache-Control响应头。因为在拦截器中添加的响应头对所有的请求都生效,并且缓存策略相同,如果想不同的请求缓存控制不同,可以通过在 构造 Request 对象时,设置 CacheControl 对象,构建个性化缓存控制策略。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

OkHttp 缓存实战 的相关文章

  • 筋斗云接口编程 / 虚拟表和视图

    虚拟表和视图 表ApiLog中有一个字段叫app xff0c 表示前端应用名 xff1a 64 ApiLog id tm addr app userId userId 如果app 61 user xff0c 则关联到User表 xff1b
  • 用zt-zip Java库进行zip文件处理

    Java 标准库本身自带java util zip包 xff0c 利用该包可以解决zip文件的处理问题 但是该包提供的功能相对底层 xff0c 想要实现zip文件的处理 xff0c 需要写一 些代码 xff0c 该包并没有封装API到调用一
  • Linux临时目录/tmp与/var/tmp

    Linux有两个公知的临时目录 xff1a tmp与 var tmp xff0c 这两个目录被用户用于存储临时性的文件 xff0c 亦经常被程读写用户存储临时性数据 两个目录没有本质上的区别 xff0c 最根本的区别仅仅是系统对其中文件清理
  • Python 日志打印

    核心概念 Python标准库自带日志模块logging xff0c logging中涉及到4个核心组件 xff0c 这些组件构建了logging体系 Logger xff1a 应用程序直接使用的接口对象 xff0c 通过logger操作完成
  • Docker指定网桥和指定网桥IP

    docker network ls NETWORK ID NAME DRIVER 7fca4eb8c647 bridge bridge 9f904ee27bf5 none null cf03ee007fb4 host host Bridge
  • Python Hash操作-MD5-SHA-HMAC

    Python标准库提供了计算数据Hash的功能 xff0c 支持许多不同的算法 xff0c 常见的MD5 SHA1 SHA256 HMAC均在其中 MD5 SHA1 SHA256在hashlib模块中 xff0c HMAC在hmac模块中
  • 离线环境下火狐浏览器Firefox完全信息迁移

    火狐浏览器Firefox是一个历史比较久的网页浏览器 xff0c 当前的火狐采用顺序数字命名的版本号演进 xff0c 迭代速度较快 xff0c Windows上可以自动升级 xff0c 升级带来了安全更新和功能特性上的变化 火狐浏览器具书签
  • 用SLF4J输出log的正确姿势

    slf4j是Java的一种Log Api xff0c 类似Apache Commons Logging 最直接的log方式 logger debug 34 Entry number 34 43 i 43 34 is 34 43 String
  • Thymeleaf消息表达式

    消息表达式用于从消息源中提取消息内容实现国际化 表达式的语法 xff1a span class token tag span class token tag span class token punctuation lt span p sp
  • Thymeleaf URL表达式

    URL在Thymleaf中是第一类公民 xff0c 有其专有的表达式语法 64 共存在2大类URL xff1a 绝对URL http www your domain相对URL xff0c 分为四类相对于页面 user login html相
  • Maven resources的include和exclude

    Maven resources plugin支持明确声明 lt directory gt 指定的资源目录中哪些资源需要处理 xff0c 哪些资源可以不被处理 lt include gt 指明需要包括的资源 xff0c 位于src my re
  • Thymeleaf条件判断

    th if th if属性求Bool值 xff0c 只有true的时候其所在的标签及该标签中的内容才会被渲染到输出结果中 lt a href 61 34 comments html 34 th href 61 34 64 product c
  • 2019 蓝桥杯省赛 A 组模拟赛(一)C. 结果填空:马的管辖 (暴力搜索)

    题目 xff1a 在中国象棋中 xff0c 马是走日字的 一个马的管辖范围指的是当前位置以及一步之内能走到的位置 xff0c 下图的绿色旗子表示马能走到的位置 如果一匹马的某个方向被蹩马脚 xff0c 它就不能往这个方向跳了 xff0c 如
  • Ubuntu下安装使用Xfce4

    安装 xff1a 代码 sudo apt get install xfce4 xfce4 taskbar plugin xfce4 taskbar plugin是我需要 xff0c 你可不用 xff0c 完整安装xfce4 的桌面环境 su
  • isdigit()函数如何判断负数

    在使用字符序列isdigt函数时 xff0c 我们会发现它无法判断负数 xff0c 如 xff1a a 61 39 2 39 39 3 39 4 5 print a 0 isdigit print a 1 isdigit 输出 xff1a
  • 用python操作浏览器的三种方式

    第一种 xff1a selenium导入浏览器驱动 xff0c 用get方法打开浏览器 xff0c 例如 xff1a import time from selenium import webdriver def mac driver 61
  • Linux-虚拟网络设备-veth pair

    基本概念 Virtual Ethernet CableBidirectional FIFOOften used to cross namespaces Linux container 中用到一个叫做veth的东西 xff0c 这是一种新的设
  • openstack-neutron-OVS agent(持续更新)

    概述 ML2Plugin的主要工作是管理虚拟网络资源 xff0c 保证数据正确无误 xff0c 具体物理设备的设置则由Agent完成 L2Agent通常运行在Hypervisor xff0c 与neutron server通过RPC通信 x
  • VUE中使用EventSource接收服务器推送事件

    Vue项目中 xff0c EventSource触发的事件中this指向变了 使用const that 61 this xff0c 然后在EventSource触发的事件中使用that if typeof EventSource 61 61
  • VNC

    一 安装tigervnc server 二 配置登录帐号 三 生成xstartup与log日志 注意 xff1a 如果没有使用vncserver来 设置密码 xff0c 则service vncserver restart 是不会成功的 这

随机推荐

  • DirectUI框架GUIFW

    前言 guifw是一款基于GDI 43 的DirectUI皮肤引擎 xff0c 借鉴了DuiLib和Qt的思想 效果预览 xff1a http download csdn net detail sllins 7707771 代码已开源 xf
  • keil 提示internal command error和no sw device

    1 使用keil烧录软件的时候 xff0c jlink stlink无法识别到芯片 需要排查的问题 1 xff09 换条线 2 xff09 是不是有程序禁用了Seral Wire xff1a 使用cubeide cubeMX xff0c 容
  • 多线程编程模式之Single Threaded Execution 模式

    一 Single Threaded Execution 模式介绍 简单的来说 xff0c Single threaded execution 模式描述了在一种多线程环境下各个线程对于公用资源的使用方式 任一时刻 xff0c 只有一个线程可以
  • NVIDIA Jetson TX2 查看系统相关+运行demo

    1 查看Jetson TX2 L4T版本 xff1a head n 1 etc nv tegra release 2 查看系统版本 xff1a cat etc lsb release 3 查看系统内核 xff1a uname a 4 查看内
  • Docker镜像迁移至新的服务器(全部数据)

    1 找到你想移动的 Docker 容器的 ID 2 提交你的变更 xff0c 并且把容器保存成镜像 xff0c 命名为 newimage docker commit span class token number 3 span a09b25
  • 配置VNC环境在windows主机访问阿里云linux服务器

    配置VNC环境在windows主机访问阿里云linux服务器 虽然作为服务器使用更多的是使用字符终端连接服务器 xff0c 进行操作 xff0c 因为图形界面很消耗性能和资源 xff0c 但有的时候使用图形界面进行操作更为便捷 xff0c
  • pythondataframe输出小结

    在使用dataframe时遇到datafram在列太多的情况下总是自动换行显示的情况 xff0c 导致数据阅读困难 xff0c 效果如下 xff1a coding utf 8 import numpy as np import pandas
  • 聊聊 Redis 为什么构建自己的简单动态字符串 SDS

    我们知道 xff0c Redis 支持字符串 哈希 列表 集合和有序集合五种基本类型 那么我们如何把图片 音频 视频或者压缩文件等二进制数据保存到 Redis 中呢 xff1f 之前在使用 Memcached 缓存这类数据时是把它们转换成
  • 聊聊 Redis 高可用之持久化AOF和RDB分析

    Redis 持久化概述 Redis 是内存数据库 xff0c 数据都是存储在内存中 xff0c 为了避免进程退出导致数据的永久丢失 xff0c 需要定期将 Redis 中的数据以某种形式把内存中的数据保存到磁盘中 xff1b 当 Redis
  • mysqldump: Got error: 1044: Access denied for user XXXX when doing LOCK TABLES

    一 报错信息 在使用mysqldump 执行远程备份数据库的时候报如下错误 xff1a mysqldump Got error span class token number 1044 span Access denied span cla
  • jmap -heap [pid]运行报:Error attaching to process: sun.jvm.hotspot.debugger.DebuggerException(不允许的操作)

    一 运行环境 操作系统 xff1a Ubuntu 5 4 0 6 Java版本 xff1a JDK8 二 执行命令 jmap heap span class token punctuation span pid号 span class to
  • chkconfig: command not found

    问题描述 在 ubuntu1 16 04 10 执行 chkconfig 命令报 chkconfig command not found 说明此服务上没有安装 chkconfig 执行如下命令进行安装 span class token fu
  • Docker 基础篇 之 安装

    一 Docker安装 查看 CentOS 内核版本 Docker 要求 CentOS 系统的内核版本高于3 10 执行如下命令查询 内核版本 span class token function uname span r span class
  • Java 基础 之 Valid 验证

    一 64 Valid 简介 Bean Validation 内置的校验器 校验器说明 64 Null被注解的元素必须为 null 64 NotNull被注解的元素必须不为 null 64 AssertTrue被注解的元素必须为 true 6
  • HttpURLConnection链接详解

    HttpURLConnection链接详解 一 简介 简单来说 xff0c HttpURLConnection 是 Java 提供的发起 HTTP 请求的基础类库 xff0c 提供了 HTTP 请求的基本功能 xff0c 不过封装的比较少
  • Apache HttpClient 详解

    1 简介 HttpClient 是 Apache Jakarta Common 下的子项目 xff0c 用来提供高效的 最新的 功能丰富的支持 HTTP 协议的客户端编程工具包 xff0c 并且它支持 HTTP 协议最新的版本和建议 Htt
  • OKHttp使用详解

    1 简介 OkHttp 是一个默认高效的 HTTP 客户端 xff1a HTTP 2 支持允许对同一主机的所有请求共享一个套接字 连接池减少了请求延迟 xff08 如果 HTTP 2 不可用 xff09 透明 GZIP 缩小了下载大小 响应
  • python二维码生成与扫码

    1 import qrcode img 61 qrcode make 34 hello world 34 img get image show img save 39 hello png 39 2 import qrcode qr 61 q
  • C语言可变参数(从stdarg.h到应用)

    1 什么是可变参数函数 在C语言编程中有时会遇到一些参数可变的函数 xff0c 例如printf scanf xff0c 其函数原型为 xff1a span class token keyword int span span class t
  • OkHttp 缓存实战

    1 简介 在实际业务中可能某些查询数据 xff0c 不经常变化 xff0c 为了节省流量 提高响应速度和增强用户体验等 xff0c 把变化频率小的数据缓存到本地 xff0c 以实现复用 OkHttp 的缓存功能使用起来也比较简单和灵活 xf