1# UIAbility组件生命周期 2 3<!--Kit: Ability Kit--> 4<!--Subsystem: Ability--> 5<!--Owner: @altay; @Luobniz21--> 6<!--Designer: @altay--> 7<!--Tester: @lixueqing513--> 8<!--Adviser: @huipeizi--> 9 10## 概述 11 12当用户在执行应用启动、应用前后台切换、应用退出等操作时,系统会触发相关应用组件的生命周期回调。其中,UIAbility组件的核心生命周期回调包括[onCreate](../reference/apis-ability-kit/js-apis-app-ability-uiAbility.md#oncreate)、[onForeground](../reference/apis-ability-kit/js-apis-app-ability-uiAbility.md#onforeground)、[onBackground](../reference/apis-ability-kit/js-apis-app-ability-uiAbility.md#onbackground)、[onDestroy](../reference/apis-ability-kit/js-apis-app-ability-uiAbility.md#ondestroy)。作为一种包含UI的应用组件,UIAbility的生命周期不可避免地与[WindowStage](../../application-dev/windowmanager/application-window-stage.md)的生命周期存在关联关系。 13 14UIAbility的生命周期示意图如下所示。 15 16 17 18以下是UIAbility启动到前台和后台两种场景说明,以及生命周期回调流程讲解。 19 20- UIAbility启动到前台,对应流程图参见上图。 21 22 1. 当用户启动一个UIAbility时,系统会依次触发onCreate()、onWindowStageCreate()、onForeground()生命周期回调。 23 24 2. 当用户跳转到其他应用(当前UIAbility切换到后台)时,系统会触发onBackground()生命周期回调。 25 26 3. 当用户再次将UIAbility切换到前台时,系统会依次触发onNewWant()、onForeground()生命周期回调。 27 28- UIAbility启动到后台,对应流程图参见下图。 29 30 1. 当用户通过[UIAbilityContext.startAbilityByCall()](../reference/apis-ability-kit/js-apis-inner-application-uiAbilityContext.md#startabilitybycall)接口启动一个UIAbility到后台时,系统会依次触发onCreate()、onBackground()(不会执行onWindowStageCreate()生命周期回调)生命周期回调。 31 32 2. 当用户将UIAbility拉到前台,系统会依次触发onNewWant()、onWindowStageCreate()、onForeground()生命周期回调。 33 34  35 36## 生命周期回调 37 38> **说明:** 39> 40> - 生命周期回调是在应用主线程执行,为了确保应用性能,建议在生命周期回调中,仅执行必要的轻量级操作。对于耗时任务,推荐采用异步处理或交由子线程执行,避免阻塞主线程。 41> - 如果需要感知UIAbility生命周期变化,开发者可以使用[ApplicationContext注册接口](../reference/apis-ability-kit/js-apis-inner-application-applicationContext.md#applicationcontextonabilitylifecycle)监听UIAbility生命周期变化。详见[监听UIAbility生命周期变化](./application-context-stage.md#监听uiability生命周期变化)。 42 43### onCreate() 44 45在首次创建UIAbility实例时,系统触发[onCreate()](../reference/apis-ability-kit/js-apis-app-ability-uiAbility.md#oncreate)回调。开发者可以在该回调中执行UIAbility整个生命周期中仅发生一次的启动逻辑。 46 47```ts 48import { AbilityConstant, UIAbility, Want } from '@kit.AbilityKit'; 49 50export default class EntryAbility extends UIAbility { 51 onCreate(want: Want, launchParam: AbilityConstant.LaunchParam): void { 52 // 执行UIAbility整个生命周期中仅发生一次的业务逻辑 53 } 54 // ... 55} 56``` 57 58### onWindowStageCreate() 59 60[UIAbility](../reference/apis-ability-kit/js-apis-app-ability-uiAbility.md)实例创建完成之后,在进入前台之前,系统会创建一个[WindowStage](../../application-dev/windowmanager/application-window-stage.md)。WindowStage创建完成后会进入[onWindowStageCreate()](../reference/apis-ability-kit/js-apis-app-ability-uiAbility.md#onwindowstagecreate)回调,开发者可以在该回调中进行UI加载、WindowStage的事件订阅。 61 62在onWindowStageCreate()回调中通过[loadContent()](../reference/apis-arkui/arkts-apis-window-Window.md#loadcontent9)方法设置应用要加载的页面,并根据需要调用[on('windowStageEvent')](../reference/apis-arkui/arkts-apis-window-WindowStage.md#onwindowstageevent9)方法订阅[WindowStage的事件](../reference/apis-arkui/arkts-apis-window-e.md#windowstageeventtype9)(获焦/失焦、切到前台/切到后台、前台可交互/前台不可交互)。 63 64> **说明:** 65> 66> - 不同开发场景下[WindowStage事件](../reference/apis-arkui/arkts-apis-window-e.md#windowstageeventtype9)的时序可能存在差异,WindowStage的相关使用请参见[窗口开发指导](../windowmanager/application-window-stage.md)。 67> - 对于不同类型的产品,当应用主窗口从前台进入后台时,UIAbility生命周期的变化也会存在差异。详见[不同设备生命周期的差异化行为](../windowmanager/window-overview.md#不同设备生命周期的差异化行为)。 68 69```ts 70import { UIAbility } from '@kit.AbilityKit'; 71import { window } from '@kit.ArkUI'; 72import { hilog } from '@kit.PerformanceAnalysisKit'; 73 74const DOMAIN_NUMBER: number = 0xFF00; 75 76export default class EntryAbility extends UIAbility { 77 // ... 78 onWindowStageCreate(windowStage: window.WindowStage): void { 79 // 设置WindowStage的事件订阅(获焦/失焦、切到前台/切到后台、前台可交互/前台不可交互) 80 try { 81 windowStage.on('windowStageEvent', (data) => { 82 let stageEventType: window.WindowStageEventType = data; 83 switch (stageEventType) { 84 case window.WindowStageEventType.SHOWN: // 切到前台 85 hilog.info(DOMAIN_NUMBER, 'testTag', `windowStage foreground.`); 86 break; 87 case window.WindowStageEventType.ACTIVE: // 获焦状态 88 hilog.info(DOMAIN_NUMBER, 'testTag', `windowStage active.`); 89 break; 90 case window.WindowStageEventType.INACTIVE: // 失焦状态 91 hilog.info(DOMAIN_NUMBER, 'testTag', `windowStage inactive.`); 92 break; 93 case window.WindowStageEventType.HIDDEN: // 切到后台 94 hilog.info(DOMAIN_NUMBER, 'testTag', `windowStage background.`); 95 break; 96 case window.WindowStageEventType.RESUMED: // 前台可交互状态 97 hilog.info(DOMAIN_NUMBER, 'testTag', `windowStage resumed.`); 98 break; 99 case window.WindowStageEventType.PAUSED: // 前台不可交互状态 100 hilog.info(DOMAIN_NUMBER, 'testTag', `windowStage paused.`); 101 break; 102 default: 103 break; 104 } 105 }); 106 } catch (exception) { 107 hilog.error(DOMAIN_NUMBER, 'testTag', 108 `Failed to enable the listener for window stage event changes. Cause: ${JSON.stringify(exception)}`); 109 } 110 hilog.info(DOMAIN_NUMBER, 'testTag', `%{public}s`, `Ability onWindowStageCreate`); 111 // 设置UI加载 112 windowStage.loadContent('pages/Index', (err, data) => { 113 // ... 114 }); 115 } 116} 117``` 118 119### onForeground() 120 121在[UIAbility](../reference/apis-ability-kit/js-apis-app-ability-uiAbility.md)切换至前台时且UIAbility的UI可见之前,系统触发[onForeground](../reference/apis-ability-kit/js-apis-app-ability-uiAbility.md#onforeground)回调。开发者可以在该回调中申请系统需要的资源,或者重新申请在`onBackground()`中释放的资源。系统回调该方法后,UIAbility实例进入前台状态,即UIAbility实例可以与用户交互的状态。UIAbility实例会一直处于这个状态,直到被某些动作打断(例如屏幕关闭、用户跳转到其他UIAbility)。 122 123例如,应用已获得地理位置权限。在UI显示之前,开发者可以在`onForeground()`回调中开启定位功能,从而获取到当前的位置信息。 124 125```ts 126import { UIAbility } from '@kit.AbilityKit'; 127 128export default class EntryAbility extends UIAbility { 129 // ... 130 131 onForeground(): void { 132 // 申请系统需要的资源,或者重新申请在onBackground()中释放的资源 133 } 134 // ... 135} 136``` 137 138 139### onBackground() 140 141在[UIAbility](../reference/apis-ability-kit/js-apis-app-ability-uiAbility.md)的UI完全不可见之后,系统触发[onBackground](../reference/apis-ability-kit/js-apis-app-ability-uiAbility.md#onbackground)回调,将UIAbility实例切换至后台状态。开发者可以在该回调中释放UI不可见时的无用资源,例如停止定位功能,以节省系统的资源消耗。 142 143`onBackground()`执行时间较短,无法提供足够的时间做一些耗时动作。请勿在该方法中执行保存用户数据或执行数据库事务等耗时操作。 144 145```ts 146import { UIAbility } from '@kit.AbilityKit'; 147 148export default class EntryAbility extends UIAbility { 149 // ... 150 151 onBackground(): void { 152 // 释放UI不可见时无用的资源 153 } 154 // ... 155} 156``` 157 158 159### onWindowStageWillDestroy() 160在[UIAbility](../reference/apis-ability-kit/js-apis-app-ability-uiAbility.md)实例销毁之前,系统触发[onWindowStageWillDestroy()](../reference/apis-ability-kit/js-apis-app-ability-uiAbility.md#onwindowstagewilldestroy12)回调。该回调在WindowStage销毁前执行,此时WindowStage可以使用。开发者可以在该回调中释放通过WindowStage获取的资源、注销WindowStage事件订阅等。 161 162```ts 163import { UIAbility } from '@kit.AbilityKit'; 164import { window } from '@kit.ArkUI'; 165import { BusinessError } from '@kit.BasicServicesKit'; 166import { hilog } from '@kit.PerformanceAnalysisKit'; 167 168const DOMAIN_NUMBER: number = 0xFF00; 169 170export default class EntryAbility extends UIAbility { 171 windowStage: window.WindowStage | undefined = undefined; 172 // ... 173 onWindowStageCreate(windowStage: window.WindowStage): void { 174 this.windowStage = windowStage; 175 // ... 176 } 177 178 onWindowStageWillDestroy(windowStage: window.WindowStage) { 179 // 释放通过windowStage对象获取的资源 180 // 在onWindowStageWillDestroy()中注销WindowStage事件订阅(获焦/失焦、切到前台/切到后台、前台可交互/前台不可交互) 181 try { 182 if (this.windowStage) { 183 this.windowStage.off('windowStageEvent'); 184 } 185 } catch (err) { 186 let code = (err as BusinessError).code; 187 let message = (err as BusinessError).message; 188 hilog.error(DOMAIN_NUMBER, 'testTag', `Failed to disable the listener for windowStageEvent. Code is ${code}, message is ${message}`); 189 } 190 } 191} 192``` 193 194### onWindowStageDestroy() 195在[UIAbility](../reference/apis-ability-kit/js-apis-app-ability-uiAbility.md)实例销毁之前,系统触发[onWindowStageDestroy()](../reference/apis-ability-kit/js-apis-app-ability-uiAbility.md#onwindowstagedestroy)回调,开发者可以在该回调中释放UI资源。该回调在WindowStage销毁后执行,此时WindowStage不可以使用。 196 197```ts 198import { UIAbility } from '@kit.AbilityKit'; 199import { window } from '@kit.ArkUI'; 200 201export default class EntryAbility extends UIAbility { 202 // ... 203 204 onWindowStageCreate(windowStage: window.WindowStage): void { 205 // 加载UI资源 206 } 207 208 onWindowStageDestroy() { 209 // 释放UI资源 210 } 211} 212``` 213 214### onDestroy() 215 216在[UIAbility](../reference/apis-ability-kit/js-apis-app-ability-uiAbility.md)实例销毁之前,系统触发[onDestroy](../reference/apis-ability-kit/js-apis-app-ability-uiAbility.md#ondestroy)回调。该回调是UIAbility接收到的最后一个生命周期回调,开发者可以在onDestroy()回调中进行系统资源的释放、数据的保存等操作。 217 218例如,开发者调用[terminateSelf()](../reference/apis-ability-kit/js-apis-inner-application-uiAbilityContext.md#terminateself)方法通知系统停止当前UIAbility实例时,系统会触发onDestroy()回调。 219<!--RP1-->再比如,用户在最近任务列表中上滑关闭UIAbility实例时,系统会触发onDestroy()回调。<!--RP1End--> 220 221```ts 222import { UIAbility } from '@kit.AbilityKit'; 223 224export default class EntryAbility extends UIAbility { 225 // ... 226 227 onDestroy() { 228 // 系统资源的释放、数据的保存等 229 } 230} 231``` 232 233### onNewWant() 234 235当应用的UIAbility实例已创建,再次调用方法启动该UIAbility实例时,系统触发该UIAbility的[onNewWant()](../reference/apis-ability-kit/js-apis-app-ability-uiAbility.md#onnewwant)回调。开发者可以在该回调中更新要加载的资源和数据等,用于后续的UI展示。 236 237```ts 238import { AbilityConstant, UIAbility, Want } from '@kit.AbilityKit'; 239 240export default class EntryAbility extends UIAbility { 241 // ... 242 243 onNewWant(want: Want, launchParam: AbilityConstant.LaunchParam) { 244 // 更新资源、数据 245 } 246} 247``` 248 249## 相关实例 250 251针对UIAbility生命周期,有以下相关实例可供参考: 252 253- [UIAbility和自定义组件生命周期(ArkTS)(API9)](https://gitee.com/openharmony/codelabs/blob/master/Ability/UIAbilityLifeCycle/README.md)