Harmony状态管理@Local和@Param

简介: Harmony状态管理@Local和@Param

Harmony 状态管理 @Local 和 @Param

@Local 背景

@Localharmony应用开发中的v2版本中 对标**@State**的状态管理修饰器,它解决了 @State 对状态变量更改的检测混乱的问题:

  1. @State 修饰的状态变量 可以是组件内部自己定义的
  2. @State 修饰的状态 也可以由外部父组件传递

这样就导致了状态数据来源不唯一,在大型项目中会引发难易检测和维护状态的问题。如以下代码:

@Entry
@Component
struct Index {
  @State num: number = 100
  build() {
    Column() {
      Text("父组件的数据 " + this.num)
      Son({ num: this.num })
      Son()
    }
    .height('100%')
    .width('100%')
  }
}
@Component
struct Son {
  @State num: number = 0
  build() {
    Column() {
      Button(`子组件 ${this.num}`)
        .onClick(() => {
          this.num++
        })
    }
  }
}

@Local 基本使用

@Local的出现就是为了解决这一类问题

  1. @Local 只能用在 @Componentv2 修饰的组件上
  2. 被**@Local**装饰的变量无法从外部初始化(无法由父组件传递进来),因此必须在组件内部进行初始化

我们对上面代码稍作修改

@Entry
@Component
struct Index {
  @State num: number = 100
  build() {
    Column() {
      Text("父组件的数据 " + this.num)
      Son({ num: this.num }) // 这里就报错啦
      Son()
    }
    .height('100%')
    .width('100%')
  }
}
@ComponentV2 // 此处调整为 @ComponentV2
struct Son {
  // 此处调整为 @Local
  @Local num: number = 0
  build() {
    Column() {
      Button(`子组件 ${this.num}`)
        .onClick(() => {
          this.num++
        })
    }
  }
}

@Local与@State对比

@State @Local
参数 无。 无。
从父组件初始化 可选。 不允许外部初始化。
观察能力 能观测变量本身以及一层的成员属性,无法深度观测。 能观测变量本身,深度观测依赖@Trace装饰器。
数据传递 可以作为数据源和子组件中状态变量同步。 可以作为数据源和子组件中状态变量同步。

@Local 特别注意

  • @Local支持观测number、boolean、string、Object、class等基本类型以及Array、Set、Map、Date等内嵌类型。
  • @Local的观测能力仅限于被装饰的变量本身。当装饰简单类型时,能够观测到对变量的赋值;当装饰对象类型时,仅能观测到对对象整体的赋值;当装饰数组类型时,能观测到数组整体以及数组元素项的变化;当装饰Array、Set、Map、Date等内嵌类型时,可以观测到通过API调用带来的变化。

@Params

@Params主要表示 子组件 接收父组件传递的数据。可以和 @Local 搭配一起使用

@Params 背景

在V1版本的状态管理修饰符中,可以用来处理 父子传参的技术有:

  1. 普通属性,不需要特别的修饰符 , 不具备单向同步
  2. @Prop 单向同步,不能监听深层次属性的改变,也不能做到双向同步
  3. @Link 可以做到双向同步,但是不能监听深层次属性的改变,而且不能直接用在 列表渲染技术 - ForEach 中
  4. @ObjectLink 可以做到双向同步,但是必须和 @Observed 搭配使用 ,而且只能用在自定义组件上

1. 普通属性

普通属性,不需要特别的修饰符 , 不具备单向同步

@Entry
@Component
struct Index {
  @State num: number = 100
  build() {
    Column() {
      // 父组件传递过去的数据
      Son({ num: this.num })
        .onClick(() => {
          this.num++
          console.log("", this.num)
        })
    }
    .height('100%')
    .width('100%')
  }
}
@Component
struct Son {
  num: number = 0
  build() {
    Column() {
      Button(`子组件 ${this.num}`)
    }
  }
}

(O~QMO1YSWQ})_ZWO[}X87N.png

2. @Prop 单向同步

@Prop 单向同步

  1. 不能监听深层次属性的改变
  2. 也不能做到双向同步

在上面代码基础上 加入**@Prop**,可以检测到基本类型数据的更新

@Component
struct Son {
  @Prop num: number = 0

但是无法检测深层次属性的改变,如

class Animal {
  dog: Dog = {
    age: 100
  }
}
class Dog {
  age: number = 100
}
@Entry
@Component
struct Index {
  @State
  animal: Animal = new Animal()
  build() {
    Column() {
      // 父组件传递过去的数据
      Son({ dog: this.animal.dog })
        .onClick(() => {
          this.animal.dog.age++
          console.log("", this.animal.dog.age)
        })
    }
    .height('100%')
    .width('100%')
  }
}
@Component
struct Son {
  @Prop dog: Dog
  build() {
    Column() {
      Button(`子组件 ${this.dog.age}`)
    }
  }
}

3. @Link 双向同步

@Link 用法和@Prop基本一致

可以做到双向同步,但是

  1. 不能监听深层次属性的改变
  2. 而且不能直接用在 列表渲染技术 - ForEach以下代码演示这个效果
class Dog {
  age: number = 100
}
@Entry
@Component
struct Index {
  @State
  dogList: Dog [] = [new Dog(), new Dog(), new Dog(), new Dog()]
  build() {
    Column() {
      ForEach(this.dogList, (item: Dog) => {
        // 此处会报错  Assigning the attribute 'item' to the '@Link' decorated attribute 'dog' is not allowed. <ArkTSCheck>
        Son({ dog: item })
          .onClick(() => {
            item.age++
            console.log("", item.age)
          })
      })
    }
    .height('100%')
    .width('100%')
  }
}
@Component
struct Son {
  @Link dog: Dog
  build() {
    Column() {
      Button(`子组件 ${this.dog.age}`)
    }
  }
}


4. @ObjectLink

@ObjectLink 可以做到双向同步,但是必须和 @Observed 搭配使用 ,而且只能用在自定义组件

小结

可以看到,如果都是使用 v1版本的这一套 父子传参的技术,是十分复杂难易直接上手使用的。

@Params 介绍

Param表示组件从外部传入的状态,使得父子组件之间的数据能够进行同步:

  • @Param装饰的变量支持本地初始化,但是不允许在组件内部直接修改变量本身。
  • 如果不本地初始化,那么必须加入 @Require
  • @Param 可以做到单向同步
  • @Param 可以检测深层次属性的修改,但是该修改在数据源上必须是整体对象的更新
  • @Params 如果也想要深度监听单个属性的修改,那么需要使用 @ObservedV2@Trace

以下代码主要演示:@Param 可以检测深层次属性的修改,但是该修改在数据源上必须是整体对象的更新

class Person {
  age: number = 100
}
@Entry
@ComponentV2
struct Index {
  @Local
  person: Person = new Person()
  build() {
    Column() {
      Son({ age: this.person.age })
        .onClick(() => {
          this.person.age++
          console.log("", this.person.age)
          if (this.person.age === 105) {
            const p = new Person()
            p.age = 200
            // 整体更新,子组件可以感知到
            this.person = p
          }
        })
    }
    .height('100%')
    .width('100%')
  }
}
@ComponentV2
struct Son {
  // 要么设置 @Require 表示父组件必须传递数据
  // 要么设置 默认值
  @Require @Param age: number
  build() {
    Column() {
      Button(`子组件 ${this.age}`)
    }
  }
}

总结

  1. @Local 可以看成是 @State的替代 ,单独表示组件内部的状态
  2. @Params 可以看成 @Prop @Link @ObjectLink的替代,更加严谨
  3. @Local 和 @Params 搭配一起使用,都只能用在 @Componentv2 修饰的自定义组件上
目录
相关文章
|
分布式计算 资源调度 Hadoop
十二、Spark的安装与部署详情(Local模式,Standalone模式,Spank on YARN模式)
十二、Spark的安装与部署详情(Local模式,Standalone模式,Spank on YARN模式)
十二、Spark的安装与部署详情(Local模式,Standalone模式,Spank on YARN模式)
|
2月前
|
开发框架 .NET Docker
【Azure 应用服务】App Service .NET Core项目在Program.cs中自定义添加的logger.LogInformation,部署到App Service上后日志不显示Log Stream中的问题
【Azure 应用服务】App Service .NET Core项目在Program.cs中自定义添加的logger.LogInformation,部署到App Service上后日志不显示Log Stream中的问题
|
5月前
|
存储
LabVIEW使用VI Package Manager(VIPM)下载和管理附加组件
LabVIEW使用VI Package Manager(VIPM)下载和管理附加组件
231 1
|
5月前
|
运维 Kubernetes 前端开发
云效产品使用报错问题之部署主机上的globalParams.sh中变量的值是base64编码过的,如何解决
本合集将整理呈现用户在使用过程中遇到的报错及其对应的解决办法,包括但不限于账户权限设置错误、项目配置不正确、代码提交冲突、构建任务执行失败、测试环境异常、需求流转阻塞等问题。阿里云云效是一站式企业级研发协同和DevOps平台,为企业提供从需求规划、开发、测试、发布到运维、运营的全流程端到端服务和工具支撑,致力于提升企业的研发效能和创新能力。
|
5月前
|
Java Shell Android开发
Rockchip系列之CAN 新增framework封装service+manager访问(3)
Rockchip系列之CAN 新增framework封装service+manager访问(3)
46 2
|
5月前
|
Shell Android开发 开发者
Android系统 自定义动态修改init.custom.rc
Android系统 自定义动态修改init.custom.rc
275 0
jira学习案例53-用useUrlQueryParam管理url状态
jira学习案例53-用useUrlQueryParam管理url状态
74 0
jira学习案例53-用useUrlQueryParam管理url状态
webpack-dev-server不是内部或外部命令,也不是可运行的程序(可用)
webpack-dev-server不是内部或外部命令,也不是可运行的程序(可用)
136 0
webpack-dev-server不是内部或外部命令,也不是可运行的程序(可用)
|
JavaScript 前端开发 测试技术
loadrunner 脚本优化-关联函数web_reg_save_param()函数详解
loadrunner 脚本优化-关联函数web_reg_save_param()函数详解
192 0
|
JavaScript 前端开发
Yii2中的资源管理的sourcePath/basePath/baseUrl分别是指的什么
Yii2中的资源管理的sourcePath/basePath/baseUrl分别是指的什么
184 0
Yii2中的资源管理的sourcePath/basePath/baseUrl分别是指的什么