事件订阅(ArkTS)
HiAppEvent提供了事件订阅接口,用于获取应用的事件。
接口说明
API接口使用说明,包括参数使用限制和具体取值范围。请参考@ohos.hiviewdfx.hiAppEvent。
订阅接口功能介绍:
| 接口名 | 描述 |
|---|---|
| addWatcher(watcher: Watcher): AppEventPackageHolder | 添加应用的事件观察者。 |
| removeWatcher(watcher: Watcher): void | 移除应用的事件观察者。 |
addWatcher接口涉及I/O操作。在对性能敏感的业务场景中,开发者应根据实际需要确定该接口是在主线程还是在子线程中调用。
如果选择在子线程中调用addWatcher,需要确保该子线程在整个接口使用周期内不会被销毁,以免影响接口的正常工作。
可参考多线程并发概述,以实现在子线程中调用接口。
打点接口功能介绍:
| 接口名 | 描述 |
|---|---|
| write(info: AppEventInfo, callback: AsyncCallback<void>): void | 应用事件异步打点方法,使用callback方式作为异步回调。 |
| write(info: AppEventInfo): Promise<void> | 应用事件异步打点方法,使用Promise方式作为异步回调。 |
write接口涉及I/O操作,执行时间通常在毫秒级别。因此,开发者应根据实际业务需求,确定该接口是在主线程还是在子线程中调用。
可参考多线程并发概述,以实现在子线程中调用接口。
事件订阅开发指导
以订阅崩溃事件(系统事件)和按钮点击事件(应用事件)为例,说明开发步骤。
-
新建一个ArkTS应用工程,编辑工程中的“entry > src > main > ets > entryability > EntryAbility.ets”文件,导入所需的依赖模块:
import { hiAppEvent, hilog } from '@kit.PerformanceAnalysisKit'; -
编辑工程中的“entry > src > main > ets > entryability > EntryAbility.ets” 文件,在onCreate函数中添加对崩溃事件、按钮点击事件的订阅。
订阅崩溃事件,采用OnReceive类型观察者的订阅方式,观察者接收到事件后会立即触发OnReceive()回调。编辑“EntryAbility.ets”文件,定义OnReceive类型观察者相关方法:
hiAppEvent.addWatcher({// 开发者可以自定义观察者名称,系统会使用名称来标识不同的观察者name: 'AppCrashWatcher',// 订阅过滤条件,这里是订阅了系统事件中的崩溃事件appEventFilters: [{domain: hiAppEvent.domain.OS,names: [hiAppEvent.event.APP_CRASH]}],// 实现onReceive回调,监听到事件后实时回调onReceive: (domain: string, appEventGroups: Array<hiAppEvent.AppEventGroup>) => {hilog.info(0x0000, 'testTag', 'AppEvents HiAppEvent success to read event with onReceive callback from ArkTS');hilog.info(0x0000, 'testTag', `domain=${domain}`);for (const eventGroup of appEventGroups) {hilog.info(0x0000, 'testTag', `AppEvents HiAppEvent eventName=${eventGroup.name}`);for (const eventInfo of eventGroup.appEventInfos) {// 开发者可以获取到崩溃事件发生的时间戳hilog.info(0x0000, 'testTag', `AppEvents HiAppEvent eventInfo.params.time=${JSON.stringify(eventInfo.params['time'])}`);// 开发者可以获取到崩溃应用的包名hilog.info(0x0000, 'testTag', `AppEvents HiAppEvent eventInfo.params.bundle_name=${JSON.stringify(eventInfo.params['bundle_name'])}`);// 开发者可以获取到崩溃事件发生时的故障日志文件hilog.info(0x0000, 'testTag', `AppEvents HiAppEvent eventInfo.params.external_log=${JSON.stringify(eventInfo.params['external_log'])}`);}}}});订阅按钮点击事件,采用OnTrigger类型观察者的订阅方式。需满足triggerCondition设置的条件,才能触发OnTrigger()回调。编辑“EntryAbility.ets”文件,定义OnTrigger类型观察者相关方法:
hiAppEvent.addWatcher({// 开发者可以自定义观察者名称,系统会使用名称来标识不同的观察者name: 'ButtonClickWatcher',// 开发者可以订阅感兴趣的应用事件,此处是订阅了按钮事件appEventFilters: [{ domain: 'button' }],// 开发者可以设置订阅回调触发的条件,此处是设置为事件打点数量满足1个triggerCondition: { row: 1 },// 开发者可以自行实现订阅回调函数,以便对订阅获取到的事件打点数据进行自定义处理onTrigger: (curRow: number, curSize: number, holder: hiAppEvent.AppEventPackageHolder) => {// 如果返回的holder对象为null,表示订阅过程发生异常。因此,在记录错误日志后直接返回if (holder == null) {hilog.error(0x0000, 'testTag', 'AppEvents HiAppEvent holder is null');return;}hilog.info(0x0000, 'testTag', 'AppEvents HiAppEvent success to read event with onTrigger callback from ArkTS');hilog.info(0x0000, 'testTag', `AppEvents HiAppEvent onTrigger: curRow=%{public}d, curSize=%{public}d`, curRow, curSize);let eventPkg: hiAppEvent.AppEventPackage | null = null;// 根据设置阈值大小(默认为1条事件)去获取订阅事件包,直到将订阅数据全部取出// 返回的事件包对象为null,表示当前订阅数据已被全部取出,此次订阅回调触发结束while ((eventPkg = holder.takeNext()) != null) {// 开发者可以对事件包中的事件打点数据进行自定义处理,此处是将事件打点数据打印在日志中hilog.info(0x0000, 'testTag', `AppEvents HiAppEvent eventPkg.packageId=%{public}d`, eventPkg.packageId);hilog.info(0x0000, 'testTag', `AppEvents HiAppEvent eventPkg.row=%{public}d`, eventPkg.row);hilog.info(0x0000, 'testTag', `AppEvents HiAppEvent eventPkg.size=%{public}d`, eventPkg.size);for (const eventInfo of eventPkg.data) {hilog.info(0x0000, 'testTag', `AppEvents HiAppEvent eventPkg.info=%{public}s`, eventInfo);}}}}); -
编辑工程中的“entry > src > main > ets > pages > Index.ets” 文件,导入依赖模块:
import { BusinessError } from '@kit.BasicServicesKit';import { hiAppEvent, hilog } from '@kit.PerformanceAnalysisKit'; -
编辑工程中的“entry > src > main > ets > pages > Index.ets” 文件,新增“WatchAppCrash ArkTS&C++”按钮触发崩溃事件;新增“writeEvent ArkTS”按钮,在按钮点击的函数中进行事件打点。示例代码如下:
触发崩溃事件。
Button('WatchAppCrash ArkTS&C++').type(ButtonType.Capsule).margin({top: 20}).backgroundColor('#0D9FFB').width('80%').height('5%').onClick(() => {// 在按钮点击函数中构造一个crash场景,触发崩溃事件let result: object = JSON.parse('');})在按钮点击的函数中进行事件打点。
Button('writeEvent ArkTS').type(ButtonType.Capsule).margin({top: 20}).backgroundColor('#0D9FFB').width('80%').height('5%').onClick(() => {// 在按钮点击函数中进行事件打点,以记录按钮点击事件let eventParams: Record<string, number> = {'clickTime': 100};let eventInfo: hiAppEvent.AppEventInfo = {// 事件领域定义domain: 'button',// 事件名称定义name: 'click',// 事件类型定义eventType: hiAppEvent.EventType.BEHAVIOR,// 事件参数定义params: eventParams,};hiAppEvent.write(eventInfo).then(() => {hilog.info(0x0000, 'testTag', `AppEvents writeEvent ArkTS success`);}).catch((err: BusinessError) => {hilog.error(0x0000, 'testTag', `AppEvents HiAppEvent err.code: ${err.code}, err.message: ${err.message}`);});})
调测验证
-
点击DevEco Studio界面中的运行按钮,运行应用工程。在应用界面中点击“WatchAppCrash ArkTS&C++”按钮,触发崩溃事件。应用退出后,重新打开应用。
-
在HiLog窗口搜索“AppEvents”关键字,查看应用处理崩溃事件数据的日志:
AppEvents HiAppEvent success to read event with onReceive callback from ArkTSAppEvents HiAppEvent eventName=APP_CRASHAppEvents HiAppEvent eventInfo.params.time=1750747995874AppEvents HiAppEvent eventInfo.params.bundle_name="com.example.txxxxx"AppEvents HiAppEvent eventInfo.params.external_log=["/data/storage/el2/log/hiappevent/APP_CRASH_1750747996042_28962.log"] -
点击DevEco Studio界面中的运行按钮,运行应用工程。在应用界面中点击“writeEvent ArkTS”按钮,触发按钮点击事件并打点。
-
在HiLog窗口搜索“AppEvents”关键字,查看应用处理按钮点击事件数据的日志:
AppEvents HiAppEvent success to read event with onTrigger callback from ArkTSAppEvents HiAppEvent onTrigger: curRow=1, curSize=121AppEvents HiAppEvent eventPkg.packageId=0AppEvents HiAppEvent eventPkg.row=1AppEvents HiAppEvent eventPkg.size=121AppEvents HiAppEvent eventPkg.info={"domain_":"button","name_":"click","type_":4,"time_":1750754529033,"tz_":"","pid_":40664,"tid_":40664,"clickTime":100}AppEvents writeEvent ArkTS success