• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1# UIAbility组件间交互(设备内)
2
3
4UIAbility是系统调度的最小单元。在设备内的功能模块之间跳转时,会涉及到启动特定的UIAbility,该UIAbility可以是应用内的其他UIAbility,也可以是其他应用的UIAbility(例如启动三方支付UIAbility)。
5
6
7本章节将从如下场景分别介绍设备内UIAbility间的交互方式。对于跨设备的应用组件交互,请参见[应用组件跨设备交互(流转)](inter-device-interaction-hop-overview.md)。
8
9
10- [启动应用内的UIAbility](#启动应用内的uiability)
11
12- [启动应用内的UIAbility并获取返回结果](#启动应用内的uiability并获取返回结果)
13
14- [启动其他应用的UIAbility](#启动其他应用的uiability)
15
16- [启动其他应用的UIAbility并获取返回结果](#启动其他应用的uiability并获取返回结果)
17
18- [启动UIAbility的指定页面](#启动uiability的指定页面)
19
20- [通过Call调用实现UIAbility交互(仅对系统应用开放)](#通过call调用实现uiability交互仅对系统应用开放)
21
22
23## 启动应用内的UIAbility
24
25当一个应用内包含多个UIAbility时,存在应用内启动UIAbility的场景。例如在支付应用中从入口UIAbility启动收付款UIAbility。
26
27假设应用中有两个UIAbility:EntryAbility和FuncAbility(可以在应用的一个Module中,也可以在的不同Module中),需要从EntryAbility的页面中启动FuncAbility。
28
291. 在EntryAbility中,通过调用startAbility()方法启动UIAbility,[want](../reference/apis/js-apis-app-ability-want.md)为UIAbility实例启动的入口参数,其中bundleName为待启动应用的Bundle名称,abilityName为待启动的UIAbility名称,moduleName在待启动的UIAbility属于不同的Module时添加,parameters为自定义信息参数。示例中的context的获取方式参见[获取UIAbility的Context属性](uiability-usage.md#获取uiability的上下文信息)。
30
31   ```ts
32   let wantInfo = {
33       deviceId: '', // deviceId为空表示本设备
34       bundleName: 'com.example.myapplication',
35       abilityName: 'FuncAbility',
36       moduleName: 'module1', // moduleName非必选
37       parameters: { // 自定义信息
38           info: '来自EntryAbility Index页面',
39       },
40   }
41   // context为调用方UIAbility的AbilityContext
42   this.context.startAbility(wantInfo).then(() => {
43       // ...
44   }).catch((err) => {
45       // ...
46   })
47   ```
48
492. 在FuncAbility的生命周期回调文件中接收EntryAbility传递过来的参数。
50
51   ```ts
52   import UIAbility from '@ohos.app.ability.UIAbility';
53   import Window from '@ohos.window';
54
55   export default class FuncAbility extends UIAbility {
56       onCreate(want, launchParam) {
57   	// 接收调用方UIAbility传过来的参数
58           let funcAbilityWant = want;
59           let info = funcAbilityWant?.parameters?.info;
60           // ...
61       }
62   }
63   ```
64
653. 在FuncAbility业务完成之后,如需要停止当前UIAbility实例,在FuncAbility中通过调用terminateSelf()方法实现。
66
67   ```ts
68   // context为需要停止的UIAbility实例的AbilityContext
69   this.context.terminateSelf((err) => {
70       // ...
71   });
72   ```
73
74
75## 启动应用内的UIAbility并获取返回结果
76
77在一个EntryAbility启动另外一个FuncAbility时,希望在被启动的FuncAbility完成相关业务后,能将结果返回给调用方。例如在应用中将入口功能和帐号登录功能分别设计为两个独立的UIAbility,在帐号登录UIAbility中完成登录操作后,需要将登录的结果返回给入口UIAbility。
78
791. 在EntryAbility中,调用startAbilityForResult()接口启动FuncAbility,异步回调中的data用于接收FuncAbility停止自身后返回给EntryAbility的信息。示例中的context的获取方式参见[获取UIAbility的Context属性](uiability-usage.md#获取uiability的上下文信息)。
80
81   ```ts
82   let wantInfo = {
83       deviceId: '', // deviceId为空表示本设备
84       bundleName: 'com.example.myapplication',
85       abilityName: 'FuncAbility',
86       moduleName: 'module1', // moduleName非必选
87       parameters: { // 自定义信息
88           info: '来自EntryAbility Index页面',
89       },
90   }
91   // context为调用方UIAbility的AbilityContext
92   this.context.startAbilityForResult(wantInfo).then((data) => {
93       // ...
94   }).catch((err) => {
95       // ...
96   })
97   ```
98
992. 在FuncAbility停止自身时,需要调用terminateSelfWithResult()方法,入参abilityResult为FuncAbility需要返回给EntryAbility的信息。
100
101   ```ts
102   const RESULT_CODE: number = 1001;
103   let abilityResult = {
104       resultCode: RESULT_CODE,
105       want: {
106           bundleName: 'com.example.myapplication',
107           abilityName: 'FuncAbility',
108           moduleName: 'module1',
109           parameters: {
110               info: '来自FuncAbility Index页面',
111           },
112       },
113   }
114   // context为被调用方UIAbility的AbilityContext
115   this.context.terminateSelfWithResult(abilityResult, (err) => {
116       // ...
117   });
118   ```
119
1203. FuncAbility停止自身后,EntryAbility通过startAbilityForResult()方法回调接收被FuncAbility返回的信息,RESULT_CODE需要与前面的数值保持一致。
121
122   ```ts
123   const RESULT_CODE: number = 1001;
124
125   // ...
126
127   // context为调用方UIAbility的AbilityContext
128   this.context.startAbilityForResult(want).then((data) => {
129       if (data?.resultCode === RESULT_CODE) {
130           // 解析被调用方UIAbility返回的信息
131           let info = data.want?.parameters?.info;
132           // ...
133       }
134   }).catch((err) => {
135       // ...
136   })
137   ```
138
139
140## 启动其他应用的UIAbility
141
142启动其他应用的UIAbility,通常用户只需要完成一个通用的操作(例如需要选择一个文档应用来查看某个文档的内容信息),推荐使用[隐式Want启动](want-overview.md#want的类型)。系统会根据调用方的want参数来识别和启动匹配到的应用UIAbility。
143
144启动UIAbility有[显式Want启动和隐式Want启动](want-overview.md)两种方式。
145
146- 显式Want启动:启动一个确定应用的UIAbility,在want参数中需要设置该应用bundleName和abilityName,当需要拉起某个明确的UIAbility时,通常使用显式Want启动方式。
147
148- 隐式Want启动:根据匹配条件由用户选择启动哪一个UIAbility,即不明确指出要启动哪一个UIAbility(abilityName参数未设置),在调用startAbility()方法时,其入参want中指定了一系列的[entities](../reference/apis/js-apis-ability-wantConstant.md#wantconstantentity)字段(表示目标UIAbility额外的类别信息,如浏览器、视频播放器)和[actions](../reference/apis/js-apis-ability-wantConstant.md#wantconstantaction)字段(表示要执行的通用操作,如查看、分享、应用详情等)等参数信息,然后由系统去分析want,并帮助找到合适的UIAbility来启动。当需要拉起其他应用的UIAbility时,开发者通常不知道用户设备中应用的安装情况,也无法确定目标应用的bundleName和abilityName,通常使用隐式Want启动方式。
149
150本章节主要讲解如何通过隐式Want启动其他应用的UIAbility。
151
1521. 将多个待匹配的文档应用安装到设备,在其对应UIAbility的module.json5配置文件中,配置skills的[entities](../reference/apis/js-apis-ability-wantConstant.md#wantconstantentity)字段和[actions](../reference/apis/js-apis-ability-wantConstant.md#wantconstantaction)字段。
153
154   ```json
155   {
156     "module": {
157       "abilities": [
158         {
159           // ...
160           "skills": [
161             {
162               "entities": [
163                 // ...
164                 "entity.system.default"
165               ],
166               "actions": [
167                 // ...
168                 "ohos.want.action.viewData"
169               ]
170             }
171           ]
172         }
173       ]
174     }
175   }
176   ```
177
1782. 在调用方want参数中的entities和action需要被包含在待匹配UIAbility的skills配置的entities和actions中。系统匹配到符合entities和actions参数条件的UIAbility后,会弹出选择框展示匹配到的UIAbility实例列表供用户选择使用。示例中的context的获取方式参见[获取UIAbility的Context属性](uiability-usage.md#获取uiability的上下文信息)。
179
180   ```ts
181   let wantInfo = {
182       deviceId: '', // deviceId为空表示本设备
183       // uncomment line below if wish to implicitly query only in the specific bundle.
184       // bundleName: 'com.example.myapplication',
185       action: 'ohos.want.action.viewData',
186       // entities can be omitted.
187       entities: ['entity.system.default'],
188   }
189
190   // context为调用方UIAbility的AbilityContext
191   this.context.startAbility(wantInfo).then(() => {
192       // ...
193   }).catch((err) => {
194       // ...
195   })
196   ```
197
198   效果示意如下图所示,点击“打开PDF文档”时,会弹出选择框供用户选择。
199
200   ![uiability-intra-device-interaction](figures/uiability-intra-device-interaction.png)
201
2023. 在文档应用使用完成之后,如需要停止当前UIAbility实例,通过调用terminateSelf()方法实现。
203
204   ```ts
205   // context为需要停止的UIAbility实例的AbilityContext
206   this.context.terminateSelf((err) => {
207       // ...
208   });
209   ```
210
211
212## 启动其他应用的UIAbility并获取返回结果
213
214当使用隐式Want启动其他应用的UIAbility并希望获取返回结果时,调用方需要使用startAbilityForResult()方法启动目标UIAbility。例如主应用中需要启动三方支付并获取支付结果。
215
2161. 在支付应用对应UIAbility的module.json5配置文件中,配置skills的[entities](../reference/apis/js-apis-ability-wantConstant.md#wantconstantentity)字段和[actions](../reference/apis/js-apis-ability-wantConstant.md#wantconstantaction)字段。
217
218   ```json
219   {
220     "module": {
221       "abilities": [
222         {
223           // ...
224           "skills": [
225             {
226               "entities": [
227                 // ...
228                 "entity.system.default"
229               ],
230               "actions": [
231                 // ...
232                 "ohos.want.action.editData"
233               ]
234             }
235           ]
236         }
237       ]
238     }
239   }
240   ```
241
2422. 调用方使用startAbilityForResult()方法启动支付应用的UIAbility,在调用方want参数中的entities和action需要被包含在待匹配UIAbility的skills配置的entities和actions中。异步回调中的data用于后续接收支付UIAbility停止自身后返回给调用方的信息。系统匹配到符合entities和actions参数条件的UIAbility后,会弹出选择框展示匹配到的UIAbility实例列表供用户选择使用。
243
244   ```ts
245   let wantInfo = {
246       deviceId: '', // deviceId为空表示本设备
247       // uncomment line below if wish to implicitly query only in the specific bundle.
248       // bundleName: 'com.example.myapplication',
249       action: 'ohos.want.action.editData',
250       // entities can be omitted.
251       entities: ['entity.system.default'],
252   }
253
254   // context为调用方UIAbility的AbilityContext
255   this.context.startAbilityForResult(wantInfo).then((data) => {
256       // ...
257   }).catch((err) => {
258       // ...
259   })
260   ```
261
2623. 在支付UIAbility完成支付之后,需要调用terminateSelfWithResult()方法实现停止自身,并将abilityResult参数信息返回给调用方。
263
264   ```ts
265   const RESULT_CODE: number = 1001;
266   let abilityResult = {
267       resultCode: RESULT_CODE,
268       want: {
269           bundleName: 'com.example.myapplication',
270           abilityName: 'EntryAbility',
271           moduleName: 'entry',
272           parameters: {
273               payResult: 'OKay',
274           },
275       },
276   }
277   // context为被调用方UIAbility的AbilityContext
278   this.context.terminateSelfWithResult(abilityResult, (err) => {
279       // ...
280   });
281   ```
282
2834. 在调用方startAbilityForResult()方法回调中接收支付应用返回的信息,RESULT_CODE需要与前面terminateSelfWithResult()返回的数值保持一致。
284
285   ```ts
286   const RESULT_CODE: number = 1001;
287
288   let want = {
289     // Want参数信息
290   };
291
292   // context为调用方UIAbility的AbilityContext
293   this.context.startAbilityForResult(want).then((data) => {
294       if (data?.resultCode === RESULT_CODE) {
295           // 解析被调用方UIAbility返回的信息
296           let payResult = data.want?.parameters?.payResult;
297           // ...
298       }
299   }).catch((err) => {
300       // ...
301   })
302   ```
303
304
305## 启动UIAbility的指定页面
306
307一个UIAbility可以对应多个页面,在不同的场景下启动该UIAbility时需要展示不同的页面,例如从一个UIAbility的页面中跳转到另外一个UIAbility时,希望启动目标UIAbility的指定页面。本文主要讲解目标UIAbility首次启动和目标UIAbility非首次启动两种启动指定页面的场景,以及在讲解启动指定页面之前会讲解到在调用方如何指定启动页面。
308
309
310### 调用方UIAbility指定启动页面
311
312调用方UIAbility启动另外一个UIAbility时,通常需要跳转到指定的页面。例如FuncAbility包含两个页面(Index对应首页,Second对应功能A页面),此时需要在传入的want参数中配置指定的页面路径信息,可以通过want中的parameters参数增加一个自定义参数传递页面跳转信息。示例中的context的获取方式参见[获取UIAbility的Context属性](uiability-usage.md#获取uiability的上下文信息)。
313
314
315```ts
316let wantInfo = {
317    deviceId: '', // deviceId为空表示本设备
318    bundleName: 'com.example.myapplication',
319    abilityName: 'FuncAbility',
320    moduleName: 'module1', // moduleName非必选
321    parameters: { // 自定义参数传递页面信息
322        router: 'funcA',
323    },
324}
325// context为调用方UIAbility的AbilityContext
326this.context.startAbility(wantInfo).then(() => {
327    // ...
328}).catch((err) => {
329    // ...
330})
331```
332
333
334### 目标UIAbility首次启动
335
336目标UIAbility首次启动时,在目标UIAbility的onWindowStageCreate()生命周期回调中,解析EntryAbility传递过来的want参数,获取到需要加载的页面信息url,传入windowStage.loadContent()方法。
337
338
339```ts
340import UIAbility from '@ohos.app.ability.UIAbility'
341import Window from '@ohos.window'
342
343export default class FuncAbility extends UIAbility {
344    funcAbilityWant;
345
346    onCreate(want, launchParam) {
347        // 接收调用方UIAbility传过来的参数
348        this.funcAbilityWant = want;
349    }
350
351    onWindowStageCreate(windowStage: Window.WindowStage) {
352        // Main window is created, set main page for this ability
353        let url = 'pages/Index';
354        if (this.funcAbilityWant?.parameters?.router) {
355            if (this.funcAbilityWant.parameters.router === 'funA') {
356                url = 'pages/Second';
357            }
358        }
359        windowStage.loadContent(url, (err, data) => {
360            // ...
361        });
362    }
363}
364```
365
366
367### 目标UIAbility非首次启动
368
369经常还会遇到一类场景,当应用A已经启动且处于主页面时,回到桌面,打开应用B,并从应用B再次启动应用A,且需要跳转到应用A的指定页面。例如联系人应用和短信应用配合使用的场景。打开短信应用主页,回到桌面,此时短信应用处于已打开状态且当前处于短信应用的主页。再打开联系人应用主页,进入联系人用户A查看详情,点击短信图标,准备给用户A发送短信,此时会再次拉起短信应用且当前处于短信应用的发送页面。
370
371![uiability_not_first_started](figures/uiability_not_first_started.png)
372
373针对以上场景,即当应用A的UIAbility实例已创建,并且处于该UIAbility实例对应的主页面中,此时,从应用B中需要再次启动应用A的该UIAbility,并且需要跳转到不同的页面,这种情况下要如何实现呢?
374
3751. 在目标UIAbility中,默认加载的是Index页面。由于当前UIAbility实例之前已经创建完成,此时会进入UIAbility的onNewWant()回调中且不会进入onCreate()和onWindowStageCreate()生命周期回调,在onNewWant()回调中解析调用方传递过来的want参数,并挂在到全局变量globalThis中,以便于后续在页面中获取。
376
377   ```ts
378   import UIAbility from '@ohos.app.ability.UIAbility'
379
380   export default class FuncAbility extends UIAbility {
381       onNewWant(want, launchParam) {
382           // 接收调用方UIAbility传过来的参数
383           globalThis.funcAbilityWant = want;
384           // ...
385       }
386   }
387   ```
388
3892. 在FuncAbility中,此时需要在Index页面中通过页面路由Router模块实现指定页面的跳转,由于此时FuncAbility对应的Index页面是处于激活状态,不会重新变量声明以及进入aboutToAppear()生命周期回调中。因此可以在Index页面的onPageShow()生命周期回调中实现页面路由跳转的功能。
390
391   ```ts
392   import router from '@ohos.router';
393
394   @Entry
395   @Component
396   struct Index {
397     onPageShow() {
398       let funcAbilityWant = globalThis.funcAbilityWant;
399       let url2 = funcAbilityWant?.parameters?.router;
400       if (url2 && url2 === 'funcA') {
401         router.replaceUrl({
402           url: 'pages/Second',
403         })
404       }
405     }
406
407     // 页面展示
408     build() {
409       // ...
410     }
411   }
412   ```
413
414> ![icon-note.gif](public_sys-resources/icon-note.gif) **说明:**
415> 当被调用方[Ability的启动模式](uiability-launch-type.md)设置为standard启动模式时,每次启动都会创建一个新的实例,那么[onNewWant()](../reference/apis/js-apis-app-ability-uiAbility.md#abilityonnewwant)回调就不会被用到。
416
417
418## 通过Call调用实现UIAbility交互(仅对系统应用开放)
419
420Call调用是UIAbility能力的扩展,它为UIAbility提供一种能够被外部调用并与外部进行通信的能力。Call调用支持前台与后台两种启动方式,使UIAbility既能被拉起到前台展示UI,也可以在后台被创建并运行。Call调用在调用方与被调用方间建立了IPC通信,因此应用开发者可通过Call调用实现不同Ability之间的数据共享。
421
422Call调用的核心接口是startAbilityByCall方法,与startAbility接口的不同之处在于:
423
424- startAbilityByCall支持前台与后台两种启动方式,而startAbility仅支持前台启动。
425
426- 调用方可使用startAbilityByCall所返回的Caller对象与被调用方进行通信,而startAbilty不具备通信能力。
427
428Call调用的使用场景主要包括:
429
430- 需要与被启动的UIAbility进行通信。
431
432- 希望被启动的UIAbility在后台运行。
433
434  **表1** Call调用相关名词解释
435
436| 名词 | 描述 |
437| -------- | -------- |
438| CallerAbility | 进行Call调用的UIAbility(调用方)。 |
439| CalleeAbility | 被Call调用的UIAbility(被调用方)。 |
440| Caller | 实际对象,由startAbilityByCall接口返回,CallerAbility可使用Caller与CalleeAbility进行通信。 |
441| Callee | 实际对象,被CalleeAbility持有,可与Caller进行通信。 |
442
443Call调用示意图如下所示。
444
445  **图1** Call调用示意图
446
447  ![call](figures/call.png)
448
449- CallerAbility调用startAbilityByCall接口获取Caller,并使用Caller对象的call方法向CalleeAbility发送数据。
450
451- CalleeAbility持有一个Callee对象,通过Callee的on方法注册回调函数,当接收到Caller发送的数据时将会调用对应的回调函数。
452
453> ![icon-note.gif](public_sys-resources/icon-note.gif) **说明:**
454> 1. 当前仅支持系统应用使用Call调用。
455>
456> 2. CalleeAbility的启动模式需要为单实例。
457>
458> 3. Call调用既支持本地(设备内)Call调用,也支持跨设备Call调用,下面介绍设备内Call调用方法。跨设备Call调用方法请参见[跨设备Call调用](hop-multi-device-collaboration.md#通过跨设备call调用实现多端协同)。
459
460
461### 接口说明
462
463Call功能主要接口如下表所示。具体的API详见[接口文档](../reference/apis/js-apis-app-ability-uiAbility.md#caller)。
464
465  **表2** Call功能主要接口
466
467| 接口名 | 描述 |
468| -------- | -------- |
469| startAbilityByCall(want: Want): Promise<Caller> | 启动指定UIAbility并获取其Caller通信接口,默认为后台启动,通过配置want可实现前台启动,详见[接口文档](../reference/apis/js-apis-inner-application-uiAbilityContext.md#abilitycontextstartabilitybycall)。AbilityContext与ServiceExtensionContext均支持该接口。 |
470| on(method: string, callback: CalleeCallBack): void | 通用组件Callee注册method对应的callback方法。 |
471| off(method: string): void | 通用组件Callee解注册method的callback方法。 |
472| call(method: string, data: rpc.Parcelable): Promise<void> | 向通用组件Callee发送约定序列化数据。 |
473| callWithResult(method: string, data: rpc.Parcelable): Promise<rpc.MessageSequence> | 向通用组件Callee发送约定序列化数据, 并将Callee返回的约定序列化数据带回。 |
474| release(): void | 释放通用组件的Caller通信接口。 |
475| on(type: "release", callback: OnReleaseCallback): void | 注册通用组件通信断开监听通知。 |
476
477设备内通过Call调用实现UIAbility交互,涉及如下两部分开发:
478
479- [创建Callee被调用端](#开发步骤创建callee被调用端)
480
481- [访问Callee被调用端](#开发步骤访问callee被调用端)
482
483
484### 开发步骤(创建Callee被调用端)
485
486在Callee被调用端,需要实现指定方法的数据接收回调函数、数据的序列化及反序列化方法。在需要接收数据期间,通过on接口注册监听,无需接收数据时通过off接口解除监听。
487
4881. 配置Ability的启动模式。
489   配置module.json5,将CalleeAbility配置为单实例"singleton"。
490
491   | Json字段 | 字段说明 |
492   | -------- | -------- |
493   | "launchType" | Ability的启动模式,设置为"singleton"类型。 |
494
495   Ability配置标签示例如下:
496
497
498   ```json
499   "abilities":[{
500     "name": ".CalleeAbility",
501     "srcEntrance": "./ets/CalleeAbility/CalleeAbility.ts",
502     "launchType": "singleton",
503     "description": "$string:CalleeAbility_desc",
504     "icon": "$media:icon",
505     "label": "$string:CalleeAbility_label",
506     "visible": true
507   }]
508   ```
509
5102. 导入UIAbility模块。
511
512   ```ts
513   import Ability from '@ohos.app.ability.UIAbility';
514   ```
515
5163. 定义约定的序列化数据。
517   调用端及被调用端发送接收的数据格式需协商一致,如下示例约定数据由number和string组成。
518
519
520   ```ts
521   export default class MyParcelable {
522       num: number = 0
523       str: string = ""
524
525       constructor(num, string) {
526           this.num = num
527           this.str = string
528       }
529
530       marshalling(messageSequence) {
531           messageSequence.writeInt(this.num)
532           messageSequence.writeString(this.str)
533           return true
534       }
535
536       unmarshalling(messageSequence) {
537           this.num = messageSequence.readInt()
538           this.str = messageSequence.readString()
539           return true
540       }
541   }
542   ```
543
5444. 实现Callee.on监听及Callee.off解除监听。
545   被调用端Callee的监听函数注册时机,取决于应用开发者。注册监听之前的数据不会被处理,取消监听之后的数据不会被处理。如下示例在Ability的onCreate注册'MSG_SEND_METHOD'监听,在onDestroy取消监听,收到序列化数据后作相应处理并返回,应用开发者根据实际需要做相应处理。具体示例代码如下:
546
547
548   ```ts
549   const TAG: string = '[CalleeAbility]';
550   const MSG_SEND_METHOD: string = 'CallSendMsg';
551
552   function sendMsgCallback(data) {
553       console.info('CalleeSortFunc called');
554
555       // 获取Caller发送的序列化数据
556       let receivedData = new MyParcelable(0, '');
557       data.readParcelable(receivedData);
558       console.info(`receiveData[${receivedData.num}, ${receivedData.str}]`);
559
560       // 作相应处理
561       // 返回序列化数据result给Caller
562       return new MyParcelable(receivedData.num + 1, `send ${receivedData.str} succeed`);
563   }
564
565   export default class CalleeAbility extends Ability {
566       onCreate(want, launchParam) {
567           try {
568               this.callee.on(MSG_SEND_METHOD, sendMsgCallback);
569           } catch (error) {
570               console.info(`${MSG_SEND_METHOD} register failed with error ${JSON.stringify(error)}`);
571           }
572       }
573
574       onDestroy() {
575           try {
576               this.callee.off(MSG_SEND_METHOD);
577           } catch (error) {
578               console.error(TAG, `${MSG_SEND_METHOD} unregister failed with error ${JSON.stringify(error)}`);
579           }
580       }
581   }
582   ```
583
584
585### 开发步骤(访问Callee被调用端)
586
5871. 导入UIAbility模块。
588
589   ```ts
590   import Ability from '@ohos.app.ability.UIAbility';
591   ```
592
5932. 获取Caller通信接口。
594   Ability的context属性实现了startAbilityByCall方法,用于获取指定通用组件的Caller通信接口。如下示例通过this.context获取Ability实例的context属性,使用startAbilityByCall拉起Callee被调用端并获取Caller通信接口,注册Caller的onRelease监听。应用开发者根据实际需要做相应处理。
595
596
597   ```ts
598   // 注册caller的release监听
599   private regOnRelease(caller) {
600       try {
601           caller.on("release", (msg) => {
602               console.info(`caller onRelease is called ${msg}`);
603           })
604           console.info('caller register OnRelease succeed');
605       } catch (error) {
606           console.info(`caller register OnRelease failed with ${error}`);
607       }
608   }
609
610   async onButtonGetCaller() {
611       try {
612           this.caller = await context.startAbilityByCall({
613               bundleName: 'com.samples.CallApplication',
614               abilityName: 'CalleeAbility'
615           })
616           if (this.caller === undefined) {
617               console.info('get caller failed')
618               return
619           }
620           console.info('get caller success')
621           this.regOnRelease(this.caller)
622       } catch (error) {
623           console.info(`get caller failed with ${error}`)
624       }
625   }
626   ```
627