The documentation you are viewing is for Dapr v1.6 which is an older version of Dapr. For up-to-date documentation, see the latest version.

状态管理概览

状态管理 API 构建块概述

介绍

使用状态管理,应用程序可以在支持的状态存储中以键/值的形式存储和查询数据。 这使您能够建立有状态的、长期运行的应用程序,可以保存和检索其状态,例如购物车或游戏的会话状态。

当使用状态管理时,你的应用程序可以利用一些功能,否则自己构建这些功能会很复杂且容易出错,例如:

  • 设置并发控制和数据一致性选项。
  • 执行批量更新操作 CRUD 包括多个事务性操作。
  • 查询和过滤键/值数据。

你的应用程序可以使用 Dapr 的状态管理 API,通过状态存储组件保存、读取和查询键/值对,如下图所示。 例如,通过使用 HTTP POST,你可以保存或查询键/值对,而通过使用 HTTP GET,你可以读取特定的键并返回其值。

特性

以下是作为状态管理 API 的一部分提供的功能:

可插拔状态存储

Dapr数据存储被建模为组件,可以在不修改你的服务代码的情况下进行替换。 请访问 支持的状态存储引擎页面查看完整列表。

可配置的状态存储行为

Dapr 允许在对于状态的操作请求中附加额外的元数据,这些元数据用以描述应如何处理该请求。 你可以附加以下:

  • 并发要求
  • 一致性要求

默认情况下,您的应用程序应该假设数据存储是最终一致的,并使用last-write-wins并发模式。

并非所有的存储引擎都一样。 为了保证应用程序的可移植性,可以了解存储引擎的元数据能力,使代码适应不同的存储能力。

并发(Concurrency)

Dapr 支持使用 ETags 的乐观并发控制(Optimistic Concurrency Control/OCC)。 当请求状态值时,Dapr 总是给返回的状态附加一个 ETag 属性。 当用户代码试图更新或删除一个状态时,它应该通过更新的请求体或删除的If-Match头附加ETag。 只有当提供的ETag与状态存储中的ETag匹配时,写操作才能成功。

Dapr之所以选择OCC,是因为在不少应用中,数据更新冲突都是很少的,因为客户端是按业务上下文自然分割的,可以对不同的数据进行操作。 然而,如果你的应用选择使用ETags,请求可能会因为不匹配的ETags而被拒绝。 建议你在代码中使用重试策略来弥补使用 ETag 时的这种冲突。

如果您的应用程序在书面请求中省略了ETags,Dapr会在处理请求时跳过ETags校验。 这与ETags的last-write-wins模式相比,基本上可以实现first-write-wins模式。

阅读API参考,了解如何设置并发选项。

一致性

Dapr 同时支持强一致性最终一致性,其中最终一致性为默认行为。

当使用强一致性时,Dapr 会等待所有副本(或指定的quorums)确认后才会确认写入请求。 当最终使用一致性时,Dapr 将在基本数据存储接受写入请求后立即返回,即使这是单个副本。

阅读API参考,了解如何设置一致性选项。

批量操作

Dapr 支持两种类型的批量操作 - bulkmulti。 您可以将几个相同类型的请求分组成批量(或批次)。 Dapr将批量操作的请求作为单个请求提交给底层数据存储。 换句话说,批量(bulk)操作不是事务性的。 另一方面,您可以将不同类型的请求分组为多操作,作为原子事务处理。

阅读 API 参考 以了解如何使用批量(bulk)选项和批次(multi)选项。

状态加密

Dapr 支持客户端对应用程序状态的自动加密,并支持密钥轮换。 这在所有 Dapr 状态存储上都受支持。 有关详细信息,请阅读 操作方法:加密应用程序状态 主题。

应用程序之间的共享状态

在共享状态方面,不同的应用程序可能有不同的需求。 例如,在一个场景中,您可能想要封装某个应用程序中的所有状态,并让 Dapr 管理您的访问权限。 在不同的场景中,您可能需要两个在相同状态下工作的应用程序能够获得和保存相同的键值(keys)。 Dapr使状态能够被隔离到一个应用程序,在应用程序之间的状态存储中共享,或者让多个应用程序在不同的状态存储中共享状态。 有关更多详细信息,请阅读 操作方法:在应用程序之间共享状态

Actor 状态

事务性状态存储可用于存储 Actor 状态。 指定 Actor 要使用哪个状态存储, 在状态存储组件的元数据部分中指定属性 actorStateStore as true Actor 状态与事务状态库中的具体计划一起储存,这样可以进行一致的查询。 Actor 状态与事务状态库中的具体计划一起储存,这样可以进行一致的查询。 只有一个单一的状态存储组件可以被用作所有角色的状态存储。 阅读 API 参考 以了解更多关于 Actor 中的状态存储 和 Actor API 参考

查询状态

有两种方法来查询状态。

  • 使用 Dapr 运行时提供的 状态管理查询API
  • 直接查询状态存储 用存储的原生SDK查询。

查询API

查询API提供了一种查询在状态存储中使用状态管理保存的键/值数据的方法,而不考虑底层数据库或存储技术。 它是一个可选的状态管理API。 使用状态管理查询API,你可以对键/值数据进行过滤、排序和分页。 有关更多详细信息,请阅读 操作方法:查询状态

直接查询状态存储

Dapr保存和检索状态值,而不进行任何转换。 您可以直接从 基础状态存储 中查询并聚合状态。 例如,要在 Redis 中获取与 app ID“myApp”相关的所有状态 key,可以使用:

KEYS "myApp*"
查询 Actor 状态

如果数据存储支持 SQL 查询,您可以使用 SQL 查询 Actor 的状态。 例如使用:

SELECT * FROM StateTable WHERE Id='<app-id>||<actor-type>||<actor-id>||<key>'

您还可以跨 Actor 实例执行聚合查询,避免 Actor 框架常见的基于回合的并发性限制。 例如,要计算所有温度计Actor的平均温度,使用:

SELECT AVG(value) FROM StateTable WHERE Id LIKE '<app-id>||<thermometer>||*||temperature'

状态生存时间(TTL)。

Dapr 允许对每个消息设置生存时间(TTL)。 这意味着应用程序可以为每个存储的状态设置生存时间,并且在过期后无法检索这些状态。

状态管理 API

状态管理API可以在 状态管理API参考 中找到,它描述了如何通过提供键来检索、保存、删除和查询状态值。

下一步