Flutter 应用加速之本地缓存管理

前言

村里的老人说:“不会写缓存器的码农不是好程序员。

今天我们就来讲讲如何编写一个简单又通用的缓存管理模块。

需求分析

根据以往经验,每一个缓存器,除了要保持被缓存数据/对象之外,还需要同时记录两个与之紧密相关的时间:

  1. 过期时间 expired —— 缓存超过了一定时间,需要更新了,但当前数据仍然有效
  2. 失效时间 deprecated —— 缓存超过了很长时间,当前数据已经无效,不能再用了

在过期时间 expired 之前,缓存数据/对象可直接使用,无需刷新;
超过了过期时间 expired,但未超过失效时间 deprecated,此时缓存数据仍然有效,可以继续使用,但需要委托一个线程去获取最新数据/对象,然后再更新本地缓存;
如果后台线程多次更新失败,当前缓存数据/对象已经严重超时,即超过了 deprecated,此时应该丢弃当前缓存数据/对象,返回空数据/对象给调用者。

模块设计

首先我们设计一个 ```CacheHolder``` 类来保存被缓存数据/对象,以及与之对应的时间信息:

import 'package:object_key/object_key.dart' show Time;


/// Holder for cache value with times in seconds
class CacheHolder <V> {
  CacheHolder(V? cacheValue, double cacheLifeSpan, {double? now})
      : _value = cacheValue, _life = cacheLifeSpan {
    now ??= Time.currentTimestamp;
    _expired = now + cacheLifeSpan;
    _deprecated = now + cacheLifeSpan * 2;
  }

  V? _value;

  final double _life;      // life span (in seconds)
  double _expired = 0;     // time to expired
  double _deprecated = 0;  // time to deprecated

  V? get value => _value;

  /// update cache value with current time in seconds
  void update(V? newValue, {double? now}) {
    _value = newValue;
    now ??= Time.currentTimestamp;
    _expired = now + _life;
    _deprecated = now + _life * 2;
  }

  /// check whether cache is alive with current time in seconds
  bool isAlive({double? now}) {
    now ??= Time.currentTimestamp;
    return now < _expired;
  }

  /// check whether cache is deprecated with current time in seconds
  bool isDeprecated({double? now}) {
    now ??= Time.currentTimestamp;
    return now > _deprecated;
  }

  /// renewal cache with a temporary life span and current time in seconds
  void renewal(double? duration, {double? now}) {
    duration ??= 120;
    now ??= Time.currentTimestamp;
    _expired = now + duration;
    _deprecated = now + _life * 2;
  }

}

该类提供 update() 和 renew() 两个函数来更新缓存信息,前者为获取到最新数据之后调用以更新数据及时间,后者仅刷新一下时间,用以推迟有效时间;
另外提供两个函数 isAlive() 和 isDeprecated(),分别用于判断是否需要更新,以及当前数据是否应该丢弃。

另外,为使 CacheHolder 能适用于任意类型数据/对象,这里使用了“泛型”类型定义。

缓存池

接下来我们需要设计一个缓冲池 ```CachePool```,用于保存同类型的 ```CacheHolder```:

import 'package:object_key/object_key.dart' show Time;
import 'holder.dart';


class CachePair <V> {
  CachePair(this.value, this.holder);

  final V? value;
  final CacheHolder<V> holder;

}


/// Pool for cache holders with keys
class CachePool <K, V> {

  final Map<K, CacheHolder<V>> _holderMap = {};

  Iterable<K> get keys => _holderMap.keys;

  /// update cache holder for key
  CacheHolder<V> update(K key, CacheHolder<V> holder) {
    _holderMap[key] = holder;
    return holder;
  }

  /// update cache value for key with timestamp in seconds
  CacheHolder<V> updateValue(K key, V? value, double life, {double? now}) =>
      update(key, CacheHolder(value, life, now: now));

  /// erase cache for key
  CachePair<V>? erase(K key, {double? now}) {
    CachePair<V>? old;
    if (now != null) {
      // get exists value before erasing
      old = fetch(key, now: now);
    }
    _holderMap.remove(key);
    return old;
  }

  /// fetch cache value & its holder
  CachePair<V>? fetch(K key, {double? now}) {
    CacheHolder<V>? holder = _holderMap[key];
    if (holder == null) {
      // holder not found
      return null;
    } else if (holder.isAlive(now: now)) {
      return CachePair(holder.value, holder);
    } else {
      // holder expired
      return CachePair(null, holder);
    }
  }

  /// clear expired cache holders
  int purge({double? now}) {
    now ??= Time.currentTimestamp;
    int count = 0;
    Iterable allKeys = keys;
    CacheHolder? holder;
    for (K key in allKeys) {
      holder = _holderMap[key];
      if (holder == null || holder.isDeprecated(now: now)) {
        // remove expired holders
        _holderMap.remove(key);
        ++count;
      }
    }
    return count;
  }

}

该缓冲池提供了 3 个接口给应用层使用:

  1. 更新缓存信息;
  2. 删除缓存信息;
  3. 获取缓存信息;

另外还提供一个 purge() 函数给缓存管理器调用,以清除已失效的 CacheHolder。

缓存管理器

最后,我们还需要设计一个缓存管理器 ```CacheManager```,去统一管理所有不同类型的 ```CachePool```:

import 'package:object_key/object_key.dart' show Time;
import 'pool.dart';


class CacheManager {
  factory CacheManager() => _instance;
  static final CacheManager _instance = CacheManager._internal();
  CacheManager._internal();

  final Map<String, dynamic> _poolMap = {};

  ///  Get pool with name
  ///
  /// @param name - pool name
  /// @param <K>  - key type
  /// @param <V>  - value type
  /// @return CachePool
  CachePool<K, V> getPool<K, V>(String name) {
    CachePool<K, V>? pool = _poolMap[name];
    if (pool == null) {
      pool = CachePool();
      _poolMap[name] = pool;
    }
    return pool;
  }

  ///  Purge all pools
  ///
  /// @param now - current time
  int purge(double? now) {
    now ??= Time.currentTimestamp;
    int count = 0;
    CachePool? pool;
    Iterable allKeys = _poolMap.keys;
    for (var key in allKeys) {
      pool = _poolMap[key];
      if (pool != null) {
        count += pool.purge(now: now);
      }
    }
    return count;
  }

}

我们这个缓存管理包括两个接口:

  1. 一个工厂方法 getPool(),用于获取/创建缓存池;
  2. 一个清除接口 purge(),供系统在适当的时候(例如系统内存不足时)调用以释放缓存空间。

至此,一个简单高效的本地缓存管理模块就写好了,下面我们来看看怎么用。

应用示例

假设我们有一个类 MetaTable,其作用是从数据库或者网络中获取 meta 信息,考虑到 I/O 的时间,以及数据解析为对象所消耗的 CPU 时间等,如果该类信息访问十分频繁,我们就需要为它加上一层缓存管理。

先来看看代码:


class MetaTable implements MetaDBI {

  @override
  Future<Meta?> getMeta(ID entity) async {
    // 从数据库中获取 meta 信息
  }

  @override
  Future<bool> saveMeta(Meta meta, ID entity) async {
    // 保存 meta 信息到数据库
  }

}

class MetaCache extends MetaTable {

  final CachePool<ID, Meta> _cache = CacheManager().getPool('meta');

  @override
  Future<Meta?> getMeta(ID entity) async {
    CachePair<Meta>? pair;
    CacheHolder<Meta>? holder;
    Meta? value;
    double now = Time.currentTimeSeconds;
    await lock();
    try {
      // 1. check memory cache
      pair = _cache.fetch(entity, now: now);
      holder = pair?.holder;
      value = pair?.value;
      if (value == null) {
        if (holder == null) {
          // not load yet, wait to load
        } else if (holder.isAlive(now: now)) {
          // value not exists
          return null;
        } else {
          // cache expired, wait to reload
          holder.renewal(128, now: now);
        }
        // 2. load from database
        value = await super.getMeta(entity);
        // update cache
        _cache.updateValue(entity, value, 36000, now: now);
      }
    } finally {
      unlock();
    }
    // OK, return cache now
    return value;
  }

  @override
  Future<bool> saveMeta(Meta meta, ID entity) async {
    _cache.updateValue(entity, meta, 36000, now: Time.currentTimeSeconds);
    return await super.saveMeta(meta, entity);
  }

}

带缓存读数据

当需要读取数据时,先通过 ```_cache.fetch()``` 检查当前缓存池中是否存在有效的值:

如果 (值存在),则 {

    直接返回该值;

}

否则检查 holder;

如果 (holder 存在且未过期),则 {

    说明确实不存在该数据,返回空值;

}

否则调用父类接口获取最新数据;

然后再更新本地缓存。

带缓存写数据

写数据就简单了,只需要在调用父类接口写数据库的同时刷新一下缓存即可。

代码引用

由于我已将这部分代码提交到了 pub.dev,所以在实际应用中,你只需要在项目工程文件 ```pubspec.yaml``` 中添加:

dependencies:

    object_key: ^0.1.1

然后在需要使用的 dart 文件头引入即可:

import 'package:object_key/object_key.dart';

全部源码

GitHub 地址:

https://github.com/moky/ObjectKey/tree/main/object_key/lib/src/mem

结语

这里向大家展示了一个简单高效的本地缓存管理模块,该模块能有效避免重复创建相同对象,同时也可避免内存泄漏等问题。

合理使用该模块,可以令你的应用程序访问数据的平均速度大幅提升,特别是在重复滚动展示大量数据的列表时,能让你的应用体验更加丝滑。

如有其他问题,可以下载登录 Tarsier 与我交流(默认通讯录i找 Albert Moky)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/713149.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Superset 二次开发之Git篇 git cherry-pick

Cherry-Pick 命令是 Git 中的一种功能&#xff0c;用于将特定的提交&#xff08;commit&#xff09;从一个分支应用到另一个分支。它允许你选择性地应用某些提交&#xff0c;而不是合并整个分支。Cherry-Pick 非常适合在需要将特定更改移植到其他分支时使用&#xff0c;例如从开…

为什么用SDE(随机微分方程)来描述扩散过程【论文精读】

为什么用SDE(随机微分方程)来描述扩散过程【论文精读】 B站视频&#xff1a;为什么用SDE(随机微分方程)来描述扩散过程 论文&#xff1a;Score-Based Generative Modeling through Stochastic Differential Equations 地址&#xff1a;https://doi.org/10.48550/arXiv.2011.13…

单调栈(续)、由斐波那契数列讲述矩阵快速降幂技巧

在这里先接上一篇文章单调栈&#xff0c;这里还有单调栈的一道题 题目一&#xff08;单调栈续&#xff09; 给定一个数组arr&#xff0c; 返回所有子数组最小值的累加和 就是一个数组&#xff0c;有很多的子数组&#xff0c;每个数组肯定有一个最小值&#xff0c;要把所有子…

享元和代理模式

文章目录 享元模式1.引出享元模式1.展示网站项目需求2.传统方案解决3.问题分析 2.享元模式1.基本介绍2.原理类图3.外部状态和内部状态4.类图5.代码实现1.AbsWebSite.java 抽象的网站2.ConcreteWebSite.java 具体的网站&#xff0c;type属性是内部状态3.WebSiteFactory.java 网站…

《C语言》动态内存管理

文章目录 一、动态内存分配二、关于动态内存开辟的函数1、malloc2、free3、calloc4、realloc 三、常见的动态内存的错误1、对NULL指针的解引用操作2、对动态开辟空间的越界访问3、对非动态开辟内存使用free释放4、释放free释放一块动态开辟的内存的一部分5、对同一块动态内存多…

Ubuntu基础-VirtualBox安装增强功能

目录 零. 前言 一. 安装 1.点击安装增强功能 2.点击光盘图标 3.复制到新文件夹 4.运行命令 5.重启系统 6.成果展示 二. 打开共享 1.共享粘贴 ​编辑2.共享文件夹 三.总结 安装步骤 打开共享粘贴功能&#xff1a; 打开共享文件夹功能&#xff1a; 零. 前言 在使用…

设计模式-代理模式Proxy(结构型)

代理模式&#xff08;Proxy&#xff09; 代理模式是一种结构型模式&#xff0c;它可以通过一个类代理另一个类的功能。代理类持有被代理类的引用地址&#xff0c;负责将请求转发给代理类&#xff0c;并且可以在转发前后做一些处理 图解 角色 抽象主题&#xff08;Subject&…

upload-labs第九关教程

upload-labs第九关教程 一、源代码分析代码审计::$DATA介绍 二、绕过分析特殊字符::$data绕过上传eval.php使用burpsuite抓包进行修改放包&#xff0c;查看是否上传成功使用中国蚁剑进行连接 一、源代码分析 代码审计 $is_upload false; $msg null; if (isset($_POST[submi…

抖音a_bogus,mstoken爬虫逆向补环境2024-06-15最新版

抖音a_bogus,mstoken爬虫逆向补环境2024-06-15最新版 接口及参数 打开网页版抖音&#xff0c;右键视频进入详情页。F12打开控制台筛选detail&#xff0c;然后刷新网页&#xff0c;找到请求。可以发现我们本次的参数目标a_bogus&#xff0c;msToken在cookie中可以获得&#xf…

无公网ip、服务器无法上网如何实现外网访问

在ipv4的大环境下&#xff0c;公网ip和车牌号一样抢手&#xff0c;一个固定公网ip价格非常昂贵&#xff0c;中小企业承担不起&#xff0c;也不愿意在上面投入&#xff1b;同时勒索病毒日益猖獗&#xff0c;企业信息化负责人为了保证数据安全性&#xff0c;干脆禁止服务器上外网…

分布式微服务: springboot底层机制实现

springboot底层机制实现 搭建SpringBoot底层机制开发环境ConfigurationBean会发生什么,并分析机制提出问题: SpringBoot 是怎么启动Tomcat, 并可以支持访问Controller源码分析: SpringApplication.run()SpringBoot的debug流程 实现SpringBoot底层机制[Tomcat启动分析 Spring容…

在向量数据库中存储多模态数据,通过文字搜索图片

在向量数据中存储多模态数据&#xff0c;通过文字搜索图片&#xff0c;Chroma 支持文字和图片&#xff0c;通过 OpenClip 模型对文字以及图片做 Embedding。本文通过 Chroma 实现一个文字搜索图片的功能。 OpenClip CLIP&#xff08;Contrastive Language-Image Pretraining&…

课设--学生成绩管理系统(一)

欢迎来到 Papicatch的博客 文章目录 &#x1f349;技术核心 &#x1f349;引言 &#x1f348;标识 &#x1f348;背景 &#x1f348;项目概述 &#x1f348; 文档概述 &#x1f349;可行性分析的前提 &#x1f348;项目的要求 &#x1f348;项目的目标 &#x1f348;…

Java入门4: 泛型和集合

Java入门4: 泛型和集合 MangoGO 芒狗狗 目录 4 泛型和集合4.1 泛型4.2 Collection4.3 List4.4 ArrayList4.5 Map4.6 HashMap4.7 Set 和 HashSet4.8 Collections参考代码4 泛型和集合 Java 使用集合来组织和管理对象,本节我们重点讲解泛型和集合。主要介绍 Collection、List、A…

C#医院体检系统源码 PEIS源码 系统核心功能、特点、类型、设备对接-PACS放射科设备对接:DR、CT、MRI、钼靶。

C#医院体检系统源码 PEIS源码 系统核心功能、特点、类型、设备对接-PACS放射科设备对接:DR、CT、MRI、钼靶。 体检系统是为体检中心、医院体检科等体检机构专门开发的全流程管理系统。该系统通过软件实现检测仪器数据的自动提取&#xff0c;内置多级医生工作台&#xff0c;旨在…

远程连接服务器的工具?

远程连接服务器工具是现代工作环境中不可或缺的工具之一。它允许用户通过网络远程访问和控制远程服务器&#xff0c;为用户提供了更加便捷和高效的工作方式。无论是远程办公、远程维护还是云计算&#xff0c;远程连接服务器工具都发挥着重要的作用。 在众多远程连接服务器工具…

LabVIEW RT在非NI硬件上的应用与分析

LabVIEW RT&#xff08;实时操作系统&#xff09;可运行在非NI&#xff08;National Instruments&#xff09;硬件上&#xff0c;如研华工控机&#xff0c;但需要满足特定硬件要求。本文从硬件要求、开发和运行差异、可靠性、稳定性、优势和成本等多角度详细分析在非NI硬件上运…

【Mac】Luminar Neo for mac(图像编辑软件)软件介绍及同类型软件比较

Luminar Neo软件介绍 Luminar Neo 是一款由 Skylum 开发的功能强大的照片编辑软件&#xff0c;专为摄影师和摄影爱好者设计。它适用于 Mac 和 Windows 平台&#xff0c;提供了一系列先进的编辑工具和功能&#xff0c;使用户能够轻松提升和优化他们的照片。以下是 Luminar Neo …

沸点 | LDBC与SIGMOD联合研讨,推动图数据库创新与标准化

当地时间6月9日&#xff0c;国际基准官方平台关联数据基准委员会&#xff08;LDBC&#xff0c;Linked Data Benchmark Council&#xff09;与SIGMOD 2024&#xff08;是全球最具国际影响力的数据管理、数据处理和数据存储领域的学术顶会之一&#xff0c;ACM SIGMOD/Big Data in…

非关系型数据库NoSQL数据层解决方案 之 redis springboot整合与读写操作 2024详解以及window版redis5.0.14下载百度网盘

redis下载安装以及基本使用 下载地址 链接&#xff1a;百度网盘 请输入提取码 提取码&#xff1a;0410 一个名对应一个数值 内存级 在内存里进行操作 准备启动 我们现在就有一个redis客户端的服务器了 我们再启动一个cmd 操作redis数据库 redis里面的基本数据类型有五种 …