鸿蒙-深度监听对象数组变化

HarmonyOS next之深度监听对象数组变化

前言:

在实际的应用开发过程中,我们会发现之前提到的装饰器存在一定的局限性。它仅仅能够观察到第一层的变化,而在开发中,应用常常会依据自身需求封装数据模型。当遇到多层嵌套的情况时,就会出现问题。例如二维数组,或者数组项是 class,又或者 class 的属性是 class,这种情况下,它们第二层的属性变化是无法被观察到的。这一不足严重影响了对复杂数据结构变化的监测。不过,有解决办法,那就是 @Observed/@ObjectLink 装饰器,它们的出现为解决多层嵌套数据结构属性变化观察问题提供了可能,能够更好地满足开发中对复杂数据变化监测的要求,保障应用开发中数据处理的准确性和有效性。

@ObjectLink和@Observed类装饰器用于在涉及嵌套对象或数组的场景中进行双向数据同步

这里对官网文档进行一个解释说明

官网文档地址:https://developer.huawei.com/consumer/cn/doc/harmonyos-guides-V5/arkts-observed-and-objectlink-V5#%E5%AF%B9%E8%B1%A1%E6%95%B0%E7%BB%84

let NextID: number = 1;

// 使用装饰器监听数组对象
@Observed
class Info {
  public id: number;
  public info: number;

  constructor(info: number) {
    this.id = NextID++;
    this.info = info;
  }
}
// ObjectLink一定要放在子组件里面
@Component
struct Child {
  // 子组件Child的@ObjectLink的类型是Info
  @ObjectLink info: Info;
  label: string = 'ViewChild';

  build() {
    Row() {
      Button(`ViewChild [${this.label}] this.info.info = ${this.info ? this.info.info : "undefined"}`)
        .width(320)
        .margin(10)
        .onClick(() => {
          this.info.info += 1;
        })
    }
  }
}

@Entry
@Component
struct Parent {
  // 即使是在函数之中请求的数据,赋值时也要使用 new ,否则监听不到函数变化
  // Parent中有@State装饰的Info[]
  @State arrA: Info[] = [new Info(0), new Info(0)];

  build() {
    Column() {
      ForEach(this.arrA,
        (item: Info) => {
          Child({ label: `#${item.id}`, info: item })
        },
        (item: Info): string => item.id.toString()
      )
      // 使用@State装饰的数组的数组项初始化@ObjectLink,其中数组项是被@Observed装饰的Info的实例
      Child({ label: `ViewChild this.arrA[first]`, info: this.arrA[0] })
      Child({ label: `ViewChild this.arrA[last]`, info: this.arrA[this.arrA.length-1] })

      Button(`ViewParent: reset array`)
        .width(320)
        .margin(10)
        .onClick(() => {
          this.arrA = [new Info(0), new Info(0)];
        })
      Button(`ViewParent: push`)
        .width(320)
        .margin(10)
        .onClick(() => {
          this.arrA.push(new Info(0))
        })
      Button(`ViewParent: shift`)
        .width(320)
        .margin(10)
        .onClick(() => {
          if (this.arrA.length > 0) {
            this.arrA.shift()
          } else {
            console.log("length <= 0")
          }
        })
      Button(`ViewParent: item property in middle`)
        .width(320)
        .margin(10)
        .onClick(() => {
          this.arrA[Math.floor(this.arrA.length / 2)].info = 10;
        })
      Button(`ViewParent: item property in middle`)
        .width(320)
        .margin(10)
        .onClick(() => {
          this.arrA[Math.floor(this.arrA.length / 2)] = new Info(11);
        })
    }
  }
}

以上就是我们对数组对象的比较完美的更新方式

鸿蒙系统中,通过HarmonyOS提供的图形API,如`DisplayManager` 和 `Surface` 来创建虚拟屏幕并获取RGBA类型的屏幕数据,通常会涉及以下几个步骤: 1. **初始化**: 首先需要获取到`DisplayManager`实例,这是管理所有屏幕设备的对象。 ```java DisplayManager displayManager = DisplayManager.getDisplayManager(); ``` 2. **获取显示信息**: 确定目标虚拟屏幕的信息,比如分辨率、色彩深度等。 ```java DisplayInfo displayInfo = displayManager.getDefaultDisplayInfo(); int width = displayInfo.width; int height = displayInfo.height; int colorFormat = displayInfo.colorFormat; // 检查是否支持RGBA ``` 3. **创建Surface**: 使用`Surface` 创建一个新的虚拟屏幕,并指定颜色格式。 ```java Surface surface = new Surface(displayManager.createDisplayConfig(width, height, colorFormat)); ``` 4. **获取像素数据**: 虽然直接从`Surface` 获取RGBA数据不是标准做法,但你可以尝试监听`Surface` 的`onBuffersPresent`事件,在该回调中获取最新的屏幕缓冲区,并转换成RGBA格式。这需要对图形缓冲的操作有一定了解。 ```java surface.setOnBuffersPresentListener(new BufferPresentCallback() { @Override public void onBuffersPresent(int[] bufferIds) { byte[] rgbaData = getRgbaFromBuffer(bufferIds[0]); // 实现将缓冲区转换为RGBA数组 // ...处理RGBA数据 } }); ``` 请注意,`getRgbaFromBuffer`是一个假设存在的方法,实际操作中你需要使用鸿蒙系统提供的API或者其他图像处理库来解析缓冲区。 **相关问题--:** 1. HarmonyOS中如何注册`BufferPresentCallback`? 2. 如何检查鸿蒙系统是否支持RGBA格式? 3. 如果没有找到直接获取RGBA数据的方法,有没有其他替代方案?
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值