ActiveData在HarmonyOS中的原理分析和运用

开发 前端 OpenHarmony
ActiveData是一个持有可被观察数据的类,ActiveData需要一个观察者对象,一般是DataObserver类的具体实现。

[[418313]]

想了解更多内容,请访问:

51CTO和华为官方合作共建的鸿蒙技术社区

https://harmonyos.51cto.com

在讲解ActiveData实现原理之前,我们有必要先了解一下两个重要的类Lifecycle以及DataObserver,这两个类在ActiveData整个运行过程中扮演了非常重要的角色。

  • Lifecycle提供了观察Ability和AbilitySlice的生命周期能力
  • DataObserver通过持有一个Lifecycle对象来观察Ability或者AbilitySlice的生命周期变化,同时DataObserver还允许ActiveData观察其生命周期变化,因此DataObserver和ActiveData相互观察,DataObserver观察ActiveData的数据变化,ActiveData观察DataObserver的生命周期变化。

ActiveData作用和特点

ActiveData是一个具有感知生命周期能力变化的数据通知类组件,非常适合在一些对数据同步性较高的场景下使用,它具有以下三个特点。

基于观察者模式:

ActiveData是一个持有可被观察数据的类,ActiveData需要一个观察者对象,一般是DataObserver类的具体实现。

感知生命周期:

ActiveData具有生命周期感知能力,目前ActiveData具有两种通知模式,一种是Ability/AbilitySlice生命周期是活跃(ACTIVE)状态时才更新数据,另一种是Ability/AbilitySlice生命周期处于任何存活状态(即只要没有被销毁)都可以更新数据。

自动解除数据订阅:

ActiveData必须配合实现了Lifecycle的对象使用。当Ability/AbilitySlice被销毁(STOP状态)后,会自动解除订阅,这在一定程度上可以避免内存泄漏等问题。

实践

1.基础用法

  1. public class MainAbilitySlice extends AbilitySlice { 
  2.     private ActiveData<String> activeData; 
  3.  
  4.     private Text mText; 
  5.  
  6.     private final DataObserver<String> dataObserver = new DataObserver<String>() { 
  7.         @Override 
  8.         public void onChanged(String s) { 
  9.             mText.setText(s); 
  10.         } 
  11.     }; 
  12.  
  13.     @Override 
  14.     public void onStart(Intent intent) { 
  15.         super.onStart(intent); 
  16.         super.setUIContent(ResourceTable.Layout_ability_main); 
  17.         activeData = new ActiveData<>(); 
  18.         dataObserver.setLifecycle(getLifecycle()); 
  19.  
  20.         mText = (Text) findComponentById(ResourceTable.Id_text_helloworld); 
  21.  
  22.         subscribe(); 
  23.     } 
  24.  
  25.     private void subscribe() { 
  26.         activeData.addObserver(dataObserver, true); 
  27.     } 
  28.  
  29.     @Override 
  30.     public void onActive() { 
  31.         super.onActive(); 
  32.         activeData.setData("New Hello World"); 
  33.     } 

运行之后的截图:

【中软国际】ActiveData在HarmonyOS中的原理分析和运用-鸿蒙HarmonyOS技术社区

从运行结果可以看出,setData调用后会立即触发onChanged回调方法

2.主线程手动调用

  1. // 添加如下代码测试DataObserver的onChanged方法是否会执行 
  2. findComponentById(ResourceTable.Id_button) 
  3.                 .setClickedListener(component -> activeData.setData("I Love China")); 

 运行结果如下:

【中软国际】ActiveData在HarmonyOS中的原理分析和运用-鸿蒙HarmonyOS技术社区

从运行结果我们可以看到,onChanged方法会一直触发,并不会因为值相同而不执行,虽然暂时看不了鸿蒙源码,但我们可以大胆猜测,鸿蒙底层维护了一个类似于版本号的标记,每次setData,该标记会自动+1,从而通过此版本号来判断data是否有变化,进而决定是否触发onChanged回调方法。

3.子线程调用

  1. @Override 
  2.    public void onActive() { 
  3.        super.onActive(); 
  4.        new Thread(() -> activeData.setData("New Hello World")).start(); 
  5.    }  

4.运行后发现没有问题,可以正常调用,说明setData方法可以在子线程调用。

  1. public class MainAbilitySlice extends AbilitySlice { 
  2.     private ActiveData<String> activeData; 
  3.     private ActiveData<String> activeData2; 
  4.  
  5.     private Text mText; 
  6.  
  7.     private final DataObserver<String> dataObserver = new DataObserver<String>() { 
  8.         @Override 
  9.         public void onChanged(String s) { 
  10.             mText.setText(s); 
  11.             System.out.println("ActiveData:---onChange:"+s); 
  12.         } 
  13.     }; 
  14.  
  15.     private final DataObserver<String> dataObserver2 = new DataObserver<String>() { 
  16.         @Override 
  17.         public void onChanged(String s) { 
  18.             mText.setText(s); 
  19.             System.out.println("ActiveData:---onChange:"+s); 
  20.         } 
  21.     }; 
  22.  
  23.     @Override 
  24.     public void onStart(Intent intent) { 
  25.         super.onStart(intent); 
  26.         super.setUIContent(ResourceTable.Layout_ability_main); 
  27.         activeData = new ActiveData<>(); 
  28.         activeData2 = new ActiveData<>(); 
  29.  
  30.         dataObserver.setLifecycle(getLifecycle()); 
  31.         dataObserver2.setLifecycle(getLifecycle()); 
  32.  
  33.         mText = (Text) findComponentById(ResourceTable.Id_text_helloworld); 
  34.         findComponentById(ResourceTable.Id_button) 
  35.                 .setClickedListener(component -> activeData.setData("I Love China")); 
  36.  
  37.         findComponentById(ResourceTable.Id_addObserver_true).setClickedListener(component -> { 
  38.             System.out.println("ActiveData:-------------"); 
  39.             Intent intent1 = new Intent(); 
  40.             Operation operation = new Intent.OperationBuilder() 
  41.                     .withDeviceId(""
  42.                     .withBundleName(getBundleName()) 
  43.                     .withAbilityName(SecondAbility.class.getName()) 
  44.                     .build(); 
  45.  
  46.             intent1.setOperation(operation); 
  47.             startAbility(intent1); 
  48.             // 此处是为了验证Ability在inActive状态的值的变化情况 
  49.             new EventHandler(EventRunner.getMainEventRunner()).postTask(() -> activeData.setData("New Hello World"), 2000); 
  50.         }); 
  51.  
  52.         findComponentById(ResourceTable.Id_addObserver_false).setClickedListener(component -> { 
  53.             System.out.println("ActiveData:-------------"); 
  54.             Intent intent1 = new Intent(); 
  55.             Operation operation = new Intent.OperationBuilder() 
  56.                     .withDeviceId(""
  57.                     .withBundleName(getBundleName()) 
  58.                     .withAbilityName(SecondAbility.class.getName()) 
  59.                     .build(); 
  60.  
  61.             intent1.setOperation(operation); 
  62.             startAbility(intent1); 
  63.             // 此处是为了验证Ability在inActive状态的值的变化情况 
  64.             new EventHandler(EventRunner.getMainEventRunner()).postTask(() -> activeData2.setData("New Hello World"), 2000); 
  65.         }); 
  66.  
  67.         subscribe(); 
  68.     } 
  69.  
  70.     private void subscribe() { 
  71.         activeData.addObserver(dataObserver, true); 
  72.         activeData2.addObserver(dataObserver, false); 
  73.     } 
  74.  
  75.     @Override 
  76.     public void onActive() { 
  77.         super.onActive(); 
  78.         System.out.println("ActiveData:---onActive"); 
  79.     } 
  80.  
  81.     @Override 
  82.     protected void onInactive() { 
  83.         super.onInactive(); 
  84.         System.out.println("ActiveData:---onInactive"); 
  85.     } 
  86.  
  87.     @Override 
  88.     protected void onBackground() { 
  89.         super.onBackground(); 
  90.         System.out.println("ActiveData:---onBackground"); 
  91.     } 
  92.  
  93.     @Override 
  94.     public void onForeground(Intent intent) { 
  95.         super.onForeground(intent); 
  96.     System.out.println("ActiveData:---onForeground"); 
  97.     } 

运行效果如下: 

从以上运行结果,可以看出addObserver(dataObserver, true/false)方法的特点,当为true是表示无论Ability/AbilitySlice处于任何生命周期状态,均会触发onChanged回调方法,当为false时表示Ability/AbilitySlice只有处于ACTIVE状态时才会触发onChanged方法。

总结

  • ActiveData内部是依靠Lifecycle来感知组件的生命周期,从而可以避免内部泄漏
  • 开发者无需维护observer对象,当Ability/AbilitySlice被销毁时,相关联的observer会被自动移除
  • 当Ability/AbilitySlice处于活跃(ACTIVE)状态时,当ActiveData数据源发生变化时onChanged方法会立即触发,去更新UI或者执行我们想要的任何操作
  • setData方法可在任意线程中去调用,开发者无需关心调用者是否在主线程中
  • setData方法即使设置同样的数据对象,onChanged方法仍然会被触发

想了解更多内容,请访问:

51CTO和华为官方合作共建的鸿蒙技术社区

https://harmonyos.51cto.com

 

责任编辑:jianghua 来源: 鸿蒙社区
相关推荐

2021-07-16 11:40:58

鸿蒙HarmonyOS应用

2021-03-25 09:58:22

鸿蒙HarmonyOS应用开发

2017-01-17 09:38:52

ZooKeeperHadoopHBase

2021-01-21 05:46:22

JavaLambda开发

2015-07-07 18:15:42

集群负载均衡云计算

2011-01-11 10:21:56

TCPIP流量整形

2016-09-12 14:33:20

javaHashMap

2023-10-27 16:15:35

鸿蒙天气服务功能

2010-04-13 12:32:38

Oracle字符集

2017-08-17 09:49:06

云存储技术运用

2015-10-13 10:16:01

数据分析淘宝地域

2021-09-17 09:30:57

鸿蒙HarmonyOS应用

2021-12-14 10:24:44

可见性网络安全零信任

2015-06-15 10:12:36

Java原理分析

2017-08-01 09:37:00

深度学习美团机器学习

2011-05-04 15:21:20

swing

2009-02-27 08:56:30

IIS.Net原理分析

2023-05-09 16:11:05

2023-12-20 17:38:44

APIhttp鸿蒙

2009-03-26 13:43:59

实现Order ByMySQL
点赞
收藏

51CTO技术栈公众号