• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1# \@Provide装饰器和\@Consume装饰器:与后代组件双向同步
2
3
4\@Provide和\@Consume,应用于与后代组件的双向数据同步,应用于状态数据在多个层级之间传递的场景。不同于上文提到的父子组件之间通过命名参数机制传递,\@Provide和\@Consume摆脱参数传递机制的束缚,实现跨层级传递。
5
6
7其中\@Provide装饰的变量是在祖先组件中,可以理解为被“提供”给后代的状态变量。\@Consume装饰的变量是在后代组件中,去“消费(绑定)”祖先组件提供的变量。
8
9
10> **说明:**
11>
12> 从API version 9开始,这两个装饰器支持在ArkTS卡片中使用。
13
14
15## 概述
16
17\@Provide/\@Consume装饰的状态变量有以下特性:
18
19- \@Provide装饰的状态变量自动对其所有后代组件可用,即该变量被“provide”给他的后代组件。由此可见,\@Provide的方便之处在于,开发者不需要多次在组件之间传递变量。
20
21- 后代通过使用\@Consume去获取\@Provide提供的变量,建立在\@Provide和\@Consume之间的双向数据同步,与\@State/\@Link不同的是,前者可以在多层级的父子组件之间传递。
22
23- \@Provide和\@Consume可以通过相同的变量名或者相同的变量别名绑定,建议类型相同,否则会发生类型隐式转换,从而导致应用行为异常。
24
25
26```ts
27// 通过相同的变量名绑定
28@Provide a: number = 0;
29@Consume a: number;
30
31// 通过相同的变量别名绑定
32@Provide('a') b: number = 0;
33@Consume('a') c: number;
34```
35
36
37\@Provide和\@Consume通过相同的变量名或者相同的变量别名绑定时,\@Provide装饰的变量和\@Consume装饰的变量是一对多的关系。不允许在同一个自定义组件内,包括其子组件中声明多个同名或者同别名的\@Provide装饰的变量,@Provide的属性名或别名需要唯一且确定,如果声明多个同名或者同别名的@Provide装饰的变量,会发生运行时报错。
38
39
40## 装饰器说明
41
42\@State的规则同样适用于\@Provide,差异为\@Provide还作为多层后代的同步源。
43
44| \@Provide变量装饰器 | 说明                                       |
45| -------------- | ---------------------------------------- |
46| 装饰器参数          | 别名:常量字符串,可选。<br/>如果指定了别名,则通过别名来绑定变量;如果未指定别名,则通过变量名绑定变量。 |
47| 同步类型           | 双向同步。<br/>从\@Provide变量到所有\@Consume变量以及相反的方向的数据同步。双向同步的操作与\@State和\@Link的组合相同。 |
48| 允许装饰的变量类型      | Object、class、string、number、boolean、enum类型,以及这些类型的数组。<br/>支持Date类型。<br/>支持类型的场景请参考[观察变化](#观察变化)。<br/>不支持any,不支持简单类型和复杂类型的联合类型,不允许使用undefined和null。<br/>必须指定类型。\@Provide变量的\@Consume变量的类型必须相同。<br/>**说明:**<br/>不支持Length、ResourceStr、ResourceColor类型,Length、ResourceStr、ResourceColor为简单类型和复杂类型的联合类型。 |
49| 被装饰变量的初始值      | 必须指定。                                    |
50
51| \@Consume变量装饰器 | 说明                                       |
52| -------------- | ---------------------------------------- |
53| 装饰器参数          | 别名:常量字符串,可选。<br/>如果提供了别名,则必须有\@Provide的变量和其有相同的别名才可以匹配成功;否则,则需要变量名相同才能匹配成功。 |
54| 同步类型           | 双向:从\@Provide变量(具体请参见\@Provide)到所有\@Consume变量,以及相反的方向。双向同步操作与\@State和\@Link的组合相同。 |
55| 允许装饰的变量类型      | Object、class、string、number、boolean、enum类型,以及这些类型的数组。<br/>支持Date类型。<br/>支持类型的场景请参考[观察变化](#观察变化)。<br/>不支持any,不允许使用undefined和null。<br/>必须指定类型。\@Provide变量和\@Consume变量的类型必须相同。<br/>**说明:**<br/>\@Consume装饰的变量,在其父节点或者祖先节点上,必须有对应的属性和别名的\@Provide装饰的变量。 |
56| 被装饰变量的初始值      | 无,禁止本地初始化。                               |
57
58
59## 变量的传递/访问规则说明
60
61
62| \@Provide传递/访问 | 说明                                       |
63| -------------- | ---------------------------------------- |
64| 从父组件初始化和更新     | 可选,允许父组件中常规变量(常规变量对@Prop赋值,只是数值的初始化,常规变量的变化不会触发UI刷新,只有状态变量才能触发UI刷新)、\@State、\@Link、\@Prop、\@Provide、\@Consume、\@ObjectLink、\@StorageLink、\@StorageProp、\@LocalStorageLink和\@LocalStorageProp装饰的变量装饰变量初始化子组件\@Provide。 |
65| 用于初始化子组件       | 允许,可用于初始化\@State、\@Link、\@Prop、\@Provide。 |
66| 和父组件同步         | 否。                                       |
67| 和后代组件同步        | 和\@Consume双向同步。                          |
68| 是否支持组件外访问      | 私有,仅可以在所属组件内访问。                          |
69
70
71  **图1** \@Provide初始化规则图示
72
73
74![zh-cn_image_0000001552614217](figures/zh-cn_image_0000001552614217.png)
75
76
77| \@Consume传递/访问 | 说明                                       |
78| -------------- | ---------------------------------------- |
79| 从父组件初始化和更新     | 禁止。通过相同的变量名和alias(别名)从\@Provide初始化。      |
80| 用于初始化子组件       | 允许,可用于初始化\@State、\@Link、\@Prop、\@Provide。 |
81| 和祖先组件同步        | 和\@Provide双向同步。                          |
82| 是否支持组件外访问      | 私有,仅可以在所属组件内访问                           |
83
84
85  **图2** \@Consume初始化规则图示
86
87
88![zh-cn_image_0000001502094666](figures/zh-cn_image_0000001502094666.png)
89
90
91## 观察变化和行为表现
92
93
94### 观察变化
95
96- 当装饰的数据类型为boolean、string、number类型时,可以观察到数值的变化。
97
98- 当装饰的数据类型为class或者Object的时候,可以观察到赋值和属性赋值的变化(属性为Object.keys(observedObject)返回的所有属性)。
99
100- 当装饰的对象是array的时候,可以观察到数组的添加、删除、更新数组单元。
101
102- 当装饰的对象是Date时,可以观察到Date整体的赋值,同时可通过调用Date的接口`setFullYear`, `setMonth`, `setDate`, `setHours`, `setMinutes`, `setSeconds`, `setMilliseconds`, `setTime`, `setUTCFullYear`, `setUTCMonth`, `setUTCDate`, `setUTCHours`, `setUTCMinutes`, `setUTCSeconds`, `setUTCMilliseconds` 更新Date的属性。
103
104```ts
105@Component
106struct CompD {
107  @Consume selectedDate: Date;
108
109  build() {
110    Column() {
111      Button(`child increase the day by 1`)
112        .onClick(() => {
113          this.selectedDate.setDate(this.selectedDate.getDate() + 1)
114        })
115      Button('child update the new date')
116        .margin(10)
117        .onClick(() => {
118          this.selectedDate = new Date('2023-09-09')
119        })
120      DatePicker({
121        start: new Date('1970-1-1'),
122        end: new Date('2100-1-1'),
123        selected: this.selectedDate
124      })
125    }
126  }
127}
128
129@Entry
130@Component
131struct CompA {
132  @Provide selectedDate: Date = new Date('2021-08-08')
133
134  build() {
135    Column() {
136      Button('parent increase the day by 1')
137        .margin(10)
138        .onClick(() => {
139          this.selectedDate.setDate(this.selectedDate.getDate() + 1)
140        })
141      Button('parent update the new date')
142        .margin(10)
143        .onClick(() => {
144          this.selectedDate = new Date('2023-07-07')
145        })
146      DatePicker({
147        start: new Date('1970-1-1'),
148        end: new Date('2100-1-1'),
149        selected: this.selectedDate
150      })
151      CompD()
152    }
153  }
154}
155```
156
157### 框架行为
158
1591. 初始渲染:
160   1. \@Provide装饰的变量会以map的形式,传递给当前\@Provide所属组件的所有子组件;
161   2. 子组件中如果使用\@Consume变量,则会在map中查找是否有该变量名/alias(别名)对应的\@Provide的变量,如果查找不到,框架会抛出JS ERROR;
162   3. 在初始化\@Consume变量时,和\@State/\@Link的流程类似,\@Consume变量会保存在map中查找到的\@Provide变量,并把自己注册给\@Provide。
163
1642. 当\@Provide装饰的数据变化时:
165   1. 通过初始渲染的步骤可知,子组件\@Consume已把自己注册给父组件。父组件\@Provide变量变更后,会遍历更新所有依赖它的系统组件(elementid)和状态变量(\@Consume);
166   2. 通知\@Consume更新后,子组件所有依赖\@Consume的系统组件(elementId)都会被通知更新。以此实现\@Provide对\@Consume状态数据同步。
167
1683. 当\@Consume装饰的数据变化时:
169
170   通过初始渲染的步骤可知,子组件\@Consume持有\@Provide的实例。在\@Consume更新后调用\@Provide的更新方法,将更新的数值同步回\@Provide,以此实现\@Consume向\@Provide的同步更新。
171
172
173## 使用场景
174
175在下面的示例是与后代组件双向同步状态\@Provide和\@Consume场景。当分别点击CompA和CompD组件内Button时,reviewVotes 的更改会双向同步在CompA和CompD中。
176
177
178
179```ts
180@Component
181struct CompD {
182  // @Consume装饰的变量通过相同的属性名绑定其祖先组件CompA内的@Provide装饰的变量
183  @Consume reviewVotes: number;
184
185  build() {
186    Column() {
187      Text(`reviewVotes(${this.reviewVotes})`)
188      Button(`reviewVotes(${this.reviewVotes}), give +1`)
189        .onClick(() => this.reviewVotes += 1)
190    }
191    .width('50%')
192  }
193}
194
195@Component
196struct CompC {
197  build() {
198    Row({ space: 5 }) {
199      CompD()
200      CompD()
201    }
202  }
203}
204
205@Component
206struct CompB {
207  build() {
208    CompC()
209  }
210}
211
212@Entry
213@Component
214struct CompA {
215  // @Provide装饰的变量reviewVotes由入口组件CompA提供其后代组件
216  @Provide reviewVotes: number = 0;
217
218  build() {
219    Column() {
220      Button(`reviewVotes(${this.reviewVotes}), give +1`)
221        .onClick(() => this.reviewVotes += 1)
222      CompB()
223    }
224  }
225}
226```
227
228
229## 常见问题
230
231### \@BuilderParam尾随闭包情况下\@Provide未定义错误
232
233在此场景下,CustomWidget执行this.builder()创建子组件CustomWidgetChild时,this指向的是HomePage。因此找不到CustomWidget的\@Provide变量,所以下面示例会报找不到\@Provide错误,和\@BuilderParam连用的时候要谨慎this的指向。
234
235错误示例:
236
237```ts
238class Tmp {
239  a: string = ''
240}
241
242@Entry
243@Component
244struct HomePage {
245  @Builder
246  builder2($$: Tmp) {
247    Text(`${$$.a}测试`)
248  }
249  build() {
250    Column() {
251      CustomWidget() {
252        CustomWidgetChild({ builder: this.builder2 })
253      }
254    }
255  }
256}
257@Component
258struct CustomWidget {
259  @Provide('a') a: string = 'abc';
260  @BuilderParam
261  builder: () => void;
262  build() {
263    Column() {
264      Button('你好').onClick((x) => {
265        if (this.a == 'ddd') {
266          this.a = 'abc';
267        }
268        else {
269          this.a = 'ddd';
270        }
271      })
272      this.builder()
273    }
274  }
275}
276@Component
277struct CustomWidgetChild {
278  @Consume('a') a: string;
279  @BuilderParam
280  builder: ($$: Tmp) => void;
281  build() {
282    Column() {
283      this.builder({ a: this.a })
284    }
285  }
286}
287```
288