当前位置: 首页 > news >正文

Flutter——数据库Drift开发详细教程(六)

目录

  • 1.视图
  • 2.视图中列的可空性
  • 3.DAO
  • 4.流查询
  • 5.高级用途
  • 6.注意事项

1.视图

也可以将SQL 视图定义 为 Dart 类。为此,请编写一个抽象类来扩展View。此示例声明了一个视图,用于读取示例中架构中某个类别中添加的待办事项数量:

abstract class CategoryTodoCount extends View {// Getters define the tables that this view is reading from.Todos get todos;Categories get categories;// Custom expressions can be given a name by defining them as a getter:.Expression<int> get itemCount => todos.id.count();Query as() =>// Views can select columns defined as expression getters on the class, or// they can reference columns from other tables.select([categories.description, itemCount]).from(categories).join([innerJoin(todos, todos.category.equalsExp(categories.id))]);
}

在 Dart 视图中,使用

  • 抽象 getter 来声明您将从中读取的表(例如TodosTable get todos)。
  • Expressiongetter 添加列:(例如itemCount => todos.id.count())。
  • as用于定义支持视图的 select 语句的重写方法。 中引用的列select可能指两种类型的列:
  • 在视图本身上定义的列(itemCount如上例所示)。
  • 在引用表上定义的列(如categories.description示例中所示)。对于这些引用,
    表中列定义中使用的类型转换器等高级漂移功能也会应用于视图的列。

当被选中时,两种类型的列都将添加到视图的数据类中。

最后,需要通过将视图包含在参数中来将其添加到数据库或访问器中 views:

(tables: [Todos, Categories], views: [CategoryTodoCount])
class MyDatabase extends _$MyDatabase {

2.视图中列的可空性

对于 Dart 定义的视图,定义为Expressiongetter 的 表达式始终可空。此行为与TypedResult.read(用于从包含自定义列的复杂 select 语句中读取结果的方法)匹配。

如果引用的列可为空,或者所选表不是来自内连接(因为null在这种情况下整个表可能都是内连接),则引用另一个表的列的列可为空。

从上面的例子来看,

  • itemCount列可为空,因为它被定义为复杂 Expression
  • description引用 的列不可categories.description为空。这是因为它引用了categories,即视图
    select 语句的主表。

3.DAO

当你有大量查询时,将它们全部放入一个类中可能会变得繁琐。你可以通过将一些查询提取到主数据库类中可用的类中来避免这种情况。考虑以下代码:

part '../Dart API/todos_dao.g.dart';// the _TodosDaoMixin will be created by drift. It contains all the necessary
// fields for the tables. The <MyDatabase> type annotation is the database class
// that should use this dao.
(tables: [Todos])
class TodosDao extends DatabaseAccessor<MyDatabase> with _$TodosDaoMixin {// this constructor is required so that the main database can create an instance// of this object.TodosDao(MyDatabase db) : super(db);Stream<List<TodoEntry>> todosInCategory(Category category) {if (category == null) {return (select(todos)..where((t) => isNull(t.category))).watch();} else {return (select(todos)..where((t) => t.category.equals(category.id))).watch();}}
}

如果我们现在将类上的注释更改MyDatabase为**@DriftDatabase(tables: [Todos, Categories], daos: [TodosDao])** 并重新运行代码生成,todosDao则可以使用生成的 getter 来访问该 dao 的实例。

4.流查询

漂移的核心特性是每个查询都可以转换为自动更新流。无论查询返回单行还是多行,无论查询是从单个表读取还是连接多个表,这都能正常工作。

基础知识¶
在drift中,可运行的查询由接口表示Selectable,该接口具有以下方法:

  • Future<List> get():运行一次查询,返回所有行。
  • Future getSingle():运行查询一次,断言它产生返回的一行。
  • Future<T?> getSingleOrNull():类似getSingle(),但允许返回null空结果集。

并且每个方法都有一个匹配的**watch()**返回流的方法:

  • Stream<List> watch():监视查询,返回所有行。
  • Stream watchSingle():监视查询,断言每次运行查询时都会报告一行。
  • Stream<T?> watchSingleOrNull():类似watchSingle(),但返回空结果集null。

用于构建查询的所有漂移 API 都会返回一个Selectable可以监视的内容:

Selectable<TodoItem> allItemsAfter(DateTime min) {return select(todoItems)..where((row) => row.createdAt.isBiggerThanValue(min));
}

无论使用哪种方法,都可以使用 创建流allItemsAfter(value).watch()。由于StreamDart 中的常见构建块,因此大多数框架都可以使用它们:

  • Flutter 中,您可以使用 以声明方式监听流StreamBuilder
  • Riverpod 可以使用 来包装流。示例应用StreamProvider中也使用了此技术。

所有漂移流在监听之后都会发出最新的结果(因此即使表从未改变,您也会收到快照,而不必合并get()watch())。

5.高级用途

除了监听查询之外,您还可以直接监听表上的更新事件:

Future<void> listenForUpdates() async {final stream = tableUpdates(TableUpdateQuery.onTable(todoItems,limitUpdateKind: UpdateKind.update,));await for (final event in stream) {print('Update on todos table: $event');}
}

请注意,整个查询流功能都是以漂移方式实现的,因此流更新是一种启发式方法,可能会比必要的更频繁地触发。您也可以手动将表标记为已更新:

void markUpdated() {notifyUpdates({TableUpdate.onTable(todoItems, kind: UpdateKind.insert)});
}

6.注意事项

虽然流对于自动获取您正在运行的任何查询的更新非常有用,但了解其功能和局限性至关重要。流查询在漂移中以启发式方法实现:对于每个活动流,漂移会跟踪其正在监听的表(该信息可从查询构建器获取)。每当通过漂移 API 进行插入、更新或删除操作时,相关查询都会重新安排并再次运行。

这意味着:

  1. 数据库的其他用途(例如原生 SQLite 客户端)不会触发流查询更新。您可以手动注入更新作为解决方法。
  2. 流查询的更新频率通常会超出其应有的水平,因为我们无法仅筛选特定行的更新。这通常不是问题,但需要注意。流查询通常应该返回相对较少的行,并且执行时的计算开销不应过大。
http://www.dtcms.com/a/191882.html

相关文章:

  • 讯联云库项目开发日志(二)AOP参数拦截
  • Profinet转Ethernet/IP网关模块通信协议适配配置
  • Ubuntu使用Docker搭建SonarQube企业版(含破解方法)
  • 奇妙协同效应,EtherNet IP与PROFINET网关优化半导体生产线
  • [原创](现代Delphi 12指南):[macOS 64bit App开发]: 注意“回车换行“的跨平台使用.
  • 【C++】Module CPP:模块化编程 Demo
  • uniapp,小程序中实现文本“展开/收起“功能的最佳实践
  • SIP协议栈--osip源码梳理
  • 安装win11硬盘分区MBR还是GPT_装win11系统分区及安装教程
  • 鸿蒙OSUniApp制作动态筛选功能的列表组件(鸿蒙系统适配版)#三方框架 #Uniapp
  • 答题pk小程序道具卡的获取与应用
  • RabbitMQ 消息模式实战:从简单队列到复杂路由(二)
  • 产线视觉检测设备技术方案:基于EFISH-SCB-RK3588/SAIL-RK3588的国产化替代赛扬N100/N150全场景技术解析
  • Android Development Roadmap
  • 数学复习笔记 10
  • 华宇TAS应用中间件与亿信华辰多款软件产品完成兼容互认证
  • CSS图片垂直居中问题解决方案
  • 基于开源AI智能名片链动2+1模式S2B2C商城小程序的“互相拆台”式宣传策略研究
  • 深入探索 OpenCV:从实时视频流到图像处理的实战指南
  • 《数字分身进化论:React Native与Flutter如何打造沉浸式虚拟形象编辑》
  • ubuntu系统 usb网卡rtl8852bu驱动安装
  • 第35周Zookkeeper+Dubbo JDK不同版本介绍
  • 开源的跨语言GUI元素理解8B大模型:AgentCPM-GUI
  • Tomcat多应用部署与静态资源路径问题全解指南
  • 深度解析IP静态的工作原理,IP静态的应用场景又哪些?
  • AWS EC2 微服务 金丝雀发布(Canary Release)方案
  • 攻击溯源技术体系:从理论架构到工程化实践的深度剖析
  • Pycharm IDEA加载大文件时报错:The file size exceeds configured limit
  • 微机控制非饱和土γ射线(CT)三轴试验系统
  • 国产化中间件 替换 nginx