鸿蒙Next状态管理V2 - @Param装饰器总结

一、概述

为增强子组件接受外部参数输入的能力,开发者可使用@Param装饰器。从API version 12开始,在@ComponentV2装饰的自定义组件中支持使用@Param装饰器。当前状态管理(V2试用版)相关功能尚未成熟,建议开发者尝鲜试用。

(一)功能特点

  1. 表示外部传入状态:使得父子组件之间的数据能够进行同步。
  2. 变量初始化:支持本地初始化,但不允许在组件内部直接修改变量本身(对于对象类型的变量,可修改其内部属性)。若不在本地初始化,则需和@Require装饰器一起使用,要求必须从外部传入初始化。
  3. 同步类型:由父到子单向同步。
  4. 接受多种数据源:可以接受任意类型的数据源,包括普通变量、状态变量、常量、函数返回值等,同时支持null、undefined以及联合类型。
  5. 触发UI刷新@Param装饰的变量变化时,会刷新该变量关联的组件。

(二)与状态管理V1版本装饰器对比

状态管理V1版本常用的可接受外部传入的装饰器(如@State、@Prop、@Link、@ObjectLink)存在使用限制多、不易区分的问题,使用不当时还会导致性能问题。例如@State仅能在初始化时获得引用,改变后无法同步;@Prop单向同步复杂类型时深拷贝性能差;@Link要求数据源也是状态变量;@ObjectLink要求属性类型为@Observed装饰的类。因此推出@Param装饰器简化父子组件传值规则。

二、装饰器说明

(一)变量装饰器参数

无参数。

(二)能否本地修改

否,修改值需使用@Event装饰器的能力(对于对象类型变量,可修改其内部属性)。

(三)同步类型

由父到子单向同步。

(四)允许装饰的变量类型

包括Objectclass、string、number、boolean、enum等基本类型以及ArrayDate、MapSet等内嵌类型,支持nullundefined以及联合类型。

(五)被装饰变量的初始值

允许本地初始化,若不在本地初始化,则需要和@Require装饰器一起使用,要求必须从外部传入初始化。当同时存在本地初始值与外部传入值时,优先使用外部传入值进行初始化。

三、变量传递规则

(一)从父组件初始化

@Param装饰的变量允许本地初始化,若无本地初始化则必须从外部传入初始化。

(二)初始化子组件

@Param装饰的变量可以初始化子组件中@Param装饰的变量。

(三)同步

@Param可以和父组件传入的状态变量数据源(即@Local@Param装饰的变量)进行同步,当数据源发生变化时,会将修改同步给子组件的@Param。

四、观察变化

(一)基本类型(boolean、string、number)

可以观察来自数据源同步的变化。例如:

@Entry
@ComponentV2
struct Index {
  @Local count: number = 0;
  @Local message: string = "Hello";
  @Local flag: boolean = false;
  build() {
    Column() {
      Text(`Local ${this.count}`)
      Text(`Local ${this.message}`)
      Text(`Local ${this.flag}`)
      Button("change Local")
      .onClick(() => {
          // 对数据源的更改会同步给子组件
          this.count++;
          this.message += " World";
          this.flag =!this.flag;
      })
      Child({
        count: this.count,
        message: this.message,
        flag: this.flag
      })
    }
  }
}
@ComponentV2
struct Child {
  @Require @Param count: number;
  @Require @Param message: string;
  @Require @Param flag: boolean;
  build() {
    Column() {
      Text(`Param ${this.count}`)
      Text(`Param ${this.message}`)
      Text(`Param ${this.flag}`)
    }
  }
}

(二)类对象

仅可以观察到对类对象整体赋值的变化,无法直接观察到对类成员属性赋值的变化,对类成员属性的观察依赖@ObservedV2@Trace装饰器。例如:

class RawObject {
  name: string;
  constructor(name: string ) {
    this.name = name;
  }
}
@ObservedV2
class ObservedObject {
  @Trace name: string;
  constructor(name: string ) {
    this.name = name;
  }
}
@Entry
@ComponentV2
struct Index {
  @Local rawObject: RawObject = new RawObject("rawObject");
  @Local observedObject: ObservedObject = new ObservedObject("observedObject");
  build() {
    Column() {
      Text(`${this.rawObject.name}`)
      Text(`${this.observedObject.name}`)
      Button("change object")
      .onClick(() => {
          // 对类对象整体的修改均能观察到
          this.rawObject = new RawObject("new rawObject");
          this.observedObject = new ObservedObject("new observedObject");
      })
      Button("change name")
      .onClick(() => {
          // @Local与@Param均不具备观察类对象属性的能力,因此对rawObject.name的修改无法观察到
          this.rawObject.name = "new rawObject name";
          // 由于ObservedObject的name属性被@Trace装饰,因此对observedObject.name的修改能被观察到
          this.observedObject.name = "new observedObject name";
      })
      Child({
        rawObject: this.rawObject,
        observedObject: this.observedObject
      })
    }
  }
}
@ComponentV2
struct Child {
  @Require @Param rawObject: RawObject;
  @Require @Param observedObject: ObservedObject;
  build() {
    Column() {
      Text(`${this.rawObject.name}`)
      Text(`${this.observedObject.name}`)
    }
  }
}

(三)简单类型数组

可以观察到数组整体或数组项的变化。例如:

@Entry
@ComponentV2
struct Index {
  @Local numArr: number[] = [1,2,3,4,5];
  @Local dimensionTwo: number[][] = [[1,2,3],[4,5,6]];
  build() {
    Column() {
      Text(`${this.numArr[0]}`)
      Text(`${this.numArr[1]}`)
      Text(`${this.numArr[2]}`)
      Text(`${this.dimensionTwo[0][0]}`)
      Text(`${this.dimensionTwo[1][1]}`)
      Button("change array item")
      .onClick(() => {
          this.numArr[0]++;
          this.numArr[1] += 2;
          this.dimensionTwo[0][0] = 0;
          this.dimensionTwo[1][1] = 0;
        })
      Button("change whole array")
      .onClick(() => {
          this.numArr = [5,4,3,2,1];
          this.dimensionTwo = [[7,8,9],[0,1,2]];
        })
      Child({
        numArr: this.numArr,
        dimensionTwo: this.dimensionTwo
      })
    }
  }
}
@ComponentV2
struct Child {
  @Require @Param numArr: number[];
  @Require @Param dimensionTwo: number[][];
  build() {
    Column() {
      Text(`${this.numArr[0]}`)
      Text(`${this.numArr[1]}`)
      Text(`${this.numArr[2]}`)
      Text(`${this.dimensionTwo[0][0]}`)
      Text(`${this.dimensionTwo[1][1]}`)
    }
  }
}

(四)嵌套类或对象数组

@Param无法观察深层对象属性的变化,对深层对象属性的观测依赖@ObservedV2@Trace装饰器。例如:

@ObservedV2
class Region {
  @Trace x: number;
  @Trace y: number;
  constructor(x: number, y: number) {
    this.x = x;
    this.y = y;
  }
}
@ObservedV2
class Info {
  @Trace region: Region;
  @Trace name: string;
  constructor(name: string, x: number, y: number ) {
    this.name = name;
    this.region = new Region(x, y);
  }
}
@Entry
@ComponentV2
struct Index {
  @Local infoArr: Info[] = [new Info("Ocean", 28, 120), new Info("Mountain", 26, 20)];
  @Local originInfo: Info = new Info("Origin", 0, 0);
  build() {
    Column() {
      ForEach(this.infoArr, (info: Info) => {
        Row() {
          Text(`name: ${info.name}`)
          Text(`region: ${info.region.x}-${info.region.y}`)
        }
      })
      Row() {
        Text(`Origin name: ${this.originInfo.name}`)
        Text(`Origin region: ${this.originInfo.region.x}-${this.originInfo.region.y}`)
      }
      Button("change infoArr item")
      .onClick(() => {
          // 由于属性name被@Trace装饰,所以能够观察到
          this.infoArr[0].name = "Win";
        })
      Button("change originInfo")
      .onClick(() => {
          // 由于变量originInfo被@Local装饰,所以能够观察到
          this.originInfo = new Info("Origin", 100, 100);
        })
      Button("change originInfo region")
      .onClick(() => {
          // 由于属性x、y被@Trace装饰,所以能够观察到
          this.originInfo.region.x = 25;
          this.originInfo.region.y = 25;
        })
    }
  }
}
@ComponentV2
struct Child {
  @Param infoArr: Info[] = [];
  @Param originInfo: Info = new Info("O", 0, 0);
  build() {
    Column() {
      ForEach(this.infoArr, (info: Info) => {
        Row() {
          Text(`name: ${info.name}`)
          Text(`region: ${info.region.x}-${info.region.y}`)
        }
      })
      Row() {
        Text(`Origin name: ${this.originInfo.name}`)
        Text(`Origin region: ${this.originInfo.region.x}-${this.originInfo.region.y}`)
      }
    }
  }
}

(五)内置类型(Array、Date、Map、Set)

  1. Array:可以观察到变量整体赋值以及通过API调用(如push、popshift、unshift、splice、copyWithin、fill、reverse、sort)带来的变化。
  2. Date:可以观察到数据源对Date整体的赋值,以及调用Date的接口(如setFullYearsetMonth、setDate、setHours、setMinutes、setSeconds、setMillisecondssetTime、setUTCFullYear、setUTCMonthsetUTCDate、setUTCHourssetUTCMinutes、setUTCSecondssetUTCMilliseconds)带来的变化。
  3. Map:可以观察到数据源对Map整体的赋值,以及调用Map的接口(如set、clear、delete)带来的变化。
  4. Set:可以观察到数据源对Set整体的赋值,以及调用Set的接口(如add、clear、delete)带来的变化。

五、限制条件

  1. @Param装饰器只能在@ComponentV2装饰器的自定义组件中使用。
  2. @Param装饰的变量表示组件外部输入,需要被初始化。支持使用本地初始值做初始化,当存在外部传入值时,将优先使用外部传入的值初始化,既不使用本地初始值,也不使用外部传入值是不允许的。
  3. @Param装饰的变量在子组件中无法进行修改(但对于对象类型变量,可修改其内部属性)。

六、使用场景

(一)从父组件到子组件变量传递与同步

@Param能够接受父组件@Local@Param传递的数据并与之变化同步。例如:

@ObservedV2
class Region {
  @Trace x: number;
  @Trace y: number;
  constructor(x: number, y: number) {
    this.x = x;
    this.y = y;
  }
}
@ObservedV2
class Info {
  @Trace name: string;
  @Trace age: number;
  @Trace region: Region;
  constructor(name: string, age: number, x: number, y: number ) {
    this.name = name;
    this.age = age;
    this.region = new Region(x, y);
  }
}
@Entry
@ComponentV2
struct Index {
  @Local infoList: Info[] = [new Info("Alice", 8, 0, 0), new Info("Barry", 10, 1, 20), new Info("Cindy", 18, 24, 40)];
  build() {
    Column() {
      ForEach(this.infoList, (info: Info) => {
        MiddleComponent({ info: info })
      })
      Button("change")
      .onClick(() => {
          this.infoList[0] = new Info("Atom", 40, 27, 90);
          this.infoList[1].name = "Bob";
          this.infoList[2].region = new Region(7, 9);
        })
    }
  }
}
@ComponentV2
struct MiddleComponent {
  @Require @Param info: Info;
  build() {
    Column() {
      Text(`name: ${this.info.name}`)
      Text(`age: ${this.info.age}`)
      SubComponent({ region: this.info.region })
    }
  }
}
@ComponentV2
struct SubComponent {
  @Require @Param region: Region;
  build() {
    Column() {
      Text(`region: ${this.region.x}-${this.region.y}`)
    }
  }
}

(二)装饰特定类型变量

  1. Date类型变量:可以观察到数据源对Date整体的赋值,以及调用Date相关接口带来的变化。
  2. Map类型变量:可以观察到数据源对Map整体的赋值,以及调用Map相关接口带来的变化。
  3. Set类型变量:可以观察到数据源对Set整体的赋值,以及调用Set相关接口带来的变化。

(三)联合类型

@Param支持null、undefined以及联合类型。例如:

@Entry
@ComponentV2
struct Index {
  @Local count: number | undefined = 0;
  build() {
    Column() {
      // 相关操作及UI展示
    }
  }
}

综上所述,@Param装饰器在鸿蒙Next状态管理V2中为父子组件间的数据传递与同步提供了更方便的机制,开发者在使用时需注意其功能特点、变量传递规则、观察变化范围、限制条件以及不同类型变量的使用场景等,以便更好地在项目中应用该装饰器进行状态管理。

?著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,029评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,238评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事?!?“怎么了?”我有些...
    开封第一讲书人阅读 159,576评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,214评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,324评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,392评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,416评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,196评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,631评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,919评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,090评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,767评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,410评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,090评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,328评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,952评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,979评论 2 351

推荐阅读更多精彩内容