news 2026/6/10 14:43:12

用 Drift 实现 Repository 无缝接入本地缓存/数据库(SWR:先快后准)

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
用 Drift 实现 Repository 无缝接入本地缓存/数据库(SWR:先快后准)

1)依赖与初始化(pubspec 思路)

常见组合(按你项目选):

  • drift

  • drift_flutter(Flutter 项目推荐)

  • sqlite3_flutter_libs(iOS/Android 自带 sqlite)

  • path_provider+path

(版本你用最新即可)

2)Drift 表结构:profiles

关键字段:updatedAtMs用来做 TTL / 过期判断

import 'package:drift/drift.dart'; class Profiles extends Table { TextColumn get id => text()(); // 主键 TextColumn get name => text()(); TextColumn get avatar => text().nullable()(); IntColumn get updatedAtMs => integer()(); // 记录更新时间(毫秒) @override Set<Column> get primaryKey => {id}; }

3)Database 定义(AppDatabase)

使用drift_flutterNativeDatabase.createInBackground最省心。

import 'dart:io'; import 'package:drift/drift.dart'; import 'package:drift/drift.dart' as drift; import 'package:drift_flutter/drift_flutter.dart'; part 'app_database.g.dart'; @DriftDatabase(tables: [Profiles], daos: [ProfileDao]) class AppDatabase extends _$AppDatabase { AppDatabase() : super(_openConnection()); @override int get schemaVersion => 1; } LazyDatabase _openConnection() { return LazyDatabase(() async { return drift_flutter.openDatabase( name: 'app.db', native: const DriftNativeOptions( shareAcrossIsolates: true, ), ); }); }

说明:

  • part 'app_database.g.dart';需要 build_runner 生成

  • 文件名你可以按你工程改,比如db.dart

4)DAO:ProfileDao(watch + get + upsert)

Repository 最喜欢 DAO 提供这几个方法。

import 'package:drift/drift.dart'; import 'app_database.dart'; part 'profile_dao.g.dart'; @DriftAccessor(tables: [Profiles]) class ProfileDao extends DatabaseAccessor<AppDatabase> with _$ProfileDaoMixin { ProfileDao(AppDatabase db) : super(db); Stream<Profile?> watchProfile(String id) { return (select(profiles)..where((t) => t.id.equals(id))) .watchSingleOrNull(); } Future<Profile?> getProfile(String id) { return (select(profiles)..where((t) => t.id.equals(id))) .getSingleOrNull(); } Future<void> upsertProfile(ProfilesCompanion data) async { await into(profiles).insertOnConflictUpdate(data); } Future<void> deleteProfile(String id) async { await (delete(profiles)..where((t) => t.id.equals(id))).go(); } Future<void> clearAll() async { await delete(profiles).go(); } }

5)Domain Model + Mapper(别省略,后期维护靠它)

Domain Model

class ProfileModel { final String id; final String name; final String? avatar; ProfileModel({required this.id, required this.name, this.avatar}); }

Mapper:Drift Row ↔ Domain

Drift 的 row 类型叫Profile(与表名 Profiles 对应),下面示例:

import 'app_database.dart'; class ProfileMapper { static ProfileModel toModel(Profile row) { return ProfileModel( id: row.id, name: row.name, avatar: row.avatar, ); } static ProfilesCompanion toCompanion(ProfileModel m) { return ProfilesCompanion.insert( id: m.id, name: m.name, avatar: Value(m.avatar), updatedAtMs: DateTime.now().millisecondsSinceEpoch, ); } }

6)Remote API(Dio 获取网络数据)

接口层只负责“拿远端”,Repository 负责策略。

abstract class ProfileApi { Future<ProfileModel> fetchProfile(String id); }

7)Repository:DB 单一事实源 + refresh 回写(推荐)

7.1 watch:页面自动更新

class ProfileRepository { final ProfileApi api; final ProfileDao dao; ProfileRepository({required this.api, required this.dao}); Stream<ProfileModel?> watchProfile(String id) { return dao.watchProfile(id).map((row) => row == null ? null : ProfileMapper.toModel(row)); } Future<void> refreshProfile(String id) async { final remote = await api.fetchProfile(id); await dao.upsertProfile(ProfileMapper.toCompanion(remote)); } }

页面使用方式(思路):

  • UI 订阅watchProfile(id)→ 立即显示 DB 数据

  • 下拉刷新调用refreshProfile(id)→ 网络成功后写 DB → UI 自动更新

8)再加一层“TTL 过期策略”(先快后准 + 后台刷新)

如果你还想:DB 有旧数据先出,再判断过期自动刷新:

class CachePolicy { final Duration ttl; CachePolicy(this.ttl); bool isExpired(int updatedAtMs) { final age = DateTime.now().millisecondsSinceEpoch - updatedAtMs; return age > ttl.inMilliseconds; } } class ProfileRepositoryWithTtl { final ProfileApi api; final ProfileDao dao; final CachePolicy policy; ProfileRepositoryWithTtl({required this.api, required this.dao, required this.policy}); Stream<ProfileModel?> watchProfile(String id) { return dao.watchProfile(id).map((row) => row == null ? null : ProfileMapper.toModel(row)); } /// 页面进入时调用一次:如果过期就后台刷新 Future<void> refreshIfExpired(String id) async { final cached = await dao.getProfile(id); if (cached == null || policy.isExpired(cached.updatedAtMs)) { await refreshProfile(id); } } Future<void> refreshProfile(String id) async { final remote = await api.fetchProfile(id); await dao.upsertProfile(ProfileMapper.toCompanion(remote)); } }

9)和 401 自动刷新 Token 如何衔接?

完全无感:
Repository 调api.fetchProfile,Dio 层的 RefreshInterceptor 处理 401。
refresh 失败就触发全局onAuthExpired,UI 统一跳登录,Repository 不管。

10)你需要生成代码(Drift 必做)

你有part '*.g.dart'的文件,需要 build:

flutter pub run build_runner build --delete-conflicting-outputs
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/10 13:15:40

AI开发者的“救命稻草“:RAG、知识库和Embedding,让大模型无所不知!

今年以来&#xff0c;AI 技术已经融入了我们的工作和生活中。我们通过 AI 问答逐渐取代了之前传统的搜索&#xff0c;有了 AI 的加持&#xff0c;我们的工作效率和生活便捷度确实提高了不少。今天&#xff0c;我们就一起来了解下 AI 技术中 RAG、知识库和 Embedding 这三门技术…

作者头像 李华
网站建设 2026/6/10 13:22:51

2025年,宝妈宝爸如何用创意设计实现带娃与远程工作双赢?

在2025年,越来越多的父母正在探索一种全新的生活工作模式:一边陪伴孩子成长,一边通过创意设计实现职业价值。 这不仅是平衡,更是融合。将父母角色、创意工作者、远程办公者和终身学习者这多重身份巧妙编织,正成为新时代家庭的新选择。 一、带娃与工作:不是平衡,而是融合…

作者头像 李华
网站建设 2026/6/10 8:10:24

当AI开始“理解”招标文件:一次对招投标信息平台的极限测试

我们进行了一次为期两个月的深度测试&#xff0c;焦点不再是平台的信息覆盖面或推送速度——这些已成为头部平台的基准线。此次评测的核心&#xff0c;是检验以立达标讯为代表的智能招投标信息平台&#xff0c;其AI能力是否已从“匹配关键词”进阶到“理解项目内涵”&#xff0…

作者头像 李华
网站建设 2026/6/10 8:12:52

基于SpringBoot的网络异常流量检测系统的设计与实现(程序+文档+讲解)

课题介绍在网络安全防护精细化、异常流量识别实时性需求升级的背景下&#xff0c;传统流量监控存在 “识别滞后、误报率高、适配性差” 的痛点&#xff0c;基于 SpringBoot 构建的网络异常流量检测系统&#xff0c;适配网络管理员、安全运维人员等角色&#xff0c;实现流量采集…

作者头像 李华
网站建设 2026/6/10 8:09:03

钉钉一口气发布超20个新品:Agent OS操作系统让AI从问答走向干活

12月23日&#xff0c;AI钉钉1.1新品发布暨生态大会在杭州举办&#xff0c;钉钉正式发布全球首个为AI打造的工作智能操作系统——Agent OS&#xff0c;由此开启“人与AI协同”的全新工作方式。据「TMT星球」了解&#xff0c;AI钉钉1.1版本名为“木兰”&#xff0c;距离钉钉发布A…

作者头像 李华
网站建设 2026/6/10 0:46:57

如何通过熊猫智汇释放数字员工的潜力?

数字员工在现代运营中扮演着至关重要的角色&#xff0c;特别是在优化业务流程、降低成本和提升效率方面。通过采纳AI销冠系统&#xff0c;数字员工能够执行自动化的客户沟通和服务&#xff0c;显著减少了人工干预。企业可以借助这一系统提升服务触达率&#xff0c;实现不间断响…

作者头像 李华