• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1# PersistentStorage:持久化存储UI状态
2
3
4前两个小节介绍的LocalStorage和AppStorage都是运行时的内存,但是在应用退出再次启动后,依然能保存选定的结果,是应用开发中十分常见的现象,这就需要用到PersistentStorage。
5
6
7PersistentStorage是应用程序中的可选单例对象。此对象的作用是持久化存储选定的AppStorage属性,以确保这些属性在应用程序重新启动时的值与应用程序关闭时的值相同。
8
9
10## 概述
11
12PersistentStorage将选定的AppStorage属性保留在设备磁盘上。应用程序通过API,以决定哪些AppStorage属性应借助PersistentStorage持久化。UI和业务逻辑不直接访问PersistentStorage中的属性,所有属性访问都是对AppStorage的访问,AppStorage中的更改会自动同步到PersistentStorage。
13
14PersistentStorage和AppStorage中的属性建立双向同步。应用开发通常通过AppStorage访问PersistentStorage,另外还有一些接口可以用于管理持久化属性,但是业务逻辑始终是通过AppStorage获取和设置属性的。
15
16
17## 限制条件
18
19持久化数据是一个相对缓慢的操作,应用程序应避免以下情况:
20
21- 持久化大型数据集。
22
23- 持久化经常变化的变量。
24
25PersistentStorage的持久化变量最好是小于2kb的数据,不要大量的数据持久化,因为PersistentStorage写入磁盘的操作是同步的,大量的数据本地化读写会同步在UI线程中执行,影响UI渲染性能。如果开发者需要存储大量的数据,建议使用数据库api。
26
27PersistentStorage和UIContext相关联,需要在[UIContext](../reference/apis/js-apis-arkui-UIContext.md#uicontext)明确的时候才可以调用,可以通过在[runScopedTask](../reference/apis/js-apis-arkui-UIContext.md#runscopedtask)里明确上下文。如果没有在UIContext明确的地方调用,将导致无法持久化数据。
28
29## 使用场景
30
31
32### 从AppStorage中访问PersistentStorage初始化的属性
33
341. 初始化PersistentStorage:
35
36   ```ts
37   PersistentStorage.persistProp('aProp', 47);
38   ```
39
402. 在AppStorage获取对应属性:
41
42   ```ts
43   AppStorage.get<number>('aProp'); // returns 47
44   ```
45
46   或在组件内部定义:
47
48
49   ```ts
50   @StorageLink('aProp') aProp: number = 48;
51   ```
52
53完整代码如下:
54
55
56```ts
57PersistentStorage.persistProp('aProp', 47);
58
59@Entry
60@Component
61struct Index {
62  @State message: string = 'Hello World'
63  @StorageLink('aProp') aProp: number = 48
64
65  build() {
66    Row() {
67      Column() {
68        Text(this.message)
69        // 应用退出时会保存当前结果。重新启动后,会显示上一次的保存结果
70        Text(`${this.aProp}`)
71          .onClick(() => {
72            this.aProp += 1;
73          })
74      }
75    }
76  }
77}
78```
79
80- 新应用安装后首次启动运行:
81  1. 调用persistProp初始化PersistentStorage,首先查询在PersistentStorage本地文件中是否存在“aProp”,查询结果为不存在,因为应用是第一次安装。
82  2. 接着查询属性“aProp”在AppStorage中是否存在,依旧不存在。
83  3. 在AppStorge中创建名为“aProp”的number类型属性,属性初始值是定义的默认值47。
84  4. PersistentStorage将属性“aProp”和值47写入磁盘,AppStorage中“aProp”对应的值和其后续的更改将被持久化。
85  5. 在Index组件中创建状态变量\@StorageLink('aProp') aProp,和AppStorage中“aProp”双向绑定,在创建的过程中会在AppStorage中查找,成功找到“aProp”,所以使用其在AppStorage找到的值47。
86
87  **图1** PersistProp初始化流程  
88
89![zh-cn_image_0000001553348833](figures/zh-cn_image_0000001553348833.png)
90
91- 触发点击事件后:
92  1. 状态变量\@StorageLink('aProp') aProp改变,触发Text组件重新刷新。
93  2. \@StorageLink装饰的变量是和AppStorage中建立双向同步的,所以\@StorageLink('aProp') aProp的变化会被同步回AppStorage中。
94  3. AppStorage中“aProp”属性的改变会同步到所有绑定该“aProp”的单向或者双向变量,在本示例中没有其他的绑定“aProp”的变量。
95  4. 因为“aProp”对应的属性已经被持久化,所以在AppStorage中“aProp”的改变会触发PersistentStorage,将新的改变写入本地磁盘。
96
97- 后续启动应用:
98  1. 执行PersistentStorage.persistProp('aProp', 47),在首先查询在PersistentStorage本地文件查询“aProp”属性,成功查询到。
99  2. 将在PersistentStorage查询到的值写入AppStorage中。
100  3. 在Index组件里,\@StorageLink绑定的“aProp”为PersistentStorage写入AppStorage中的值,即为上一次退出引用存入的值。
101
102
103### 在PersistentStorage之前访问AppStorage中的属性
104
105该示例为反例。在调用PersistentStorage.persistProp或者persistProps之前使用接口访问AppStorage中的属性是错误的,因为这样的调用顺序会丢失上一次应用程序运行中的属性值:
106
107
108```ts
109let aProp = AppStorage.setOrCreate('aProp', 47);
110PersistentStorage.persistProp('aProp', 48);
111```
112
113应用在非首次运行时,先执行AppStorage.setOrCreate('aProp', 47):属性“aProp”在AppStorage中创建,其类型为number,其值设置为指定的默认值47。'aProp'是持久化的属性,所以会被写回PersistentStorage磁盘中,PersistentStorage存储的上次退出应用的值丢失。
114
115PersistentStorage.persistProp('aProp', 48):在PersistentStorage中查找到“aProp”,找到,值为47。
116