鸿蒙开发之状态管理@Observed和@ObjectLink

本文介绍了在React中,当对象内引用或数组中的对象属性变化时,常规状态管理工具无法自动更新UI的问题,展示了如何使用@Observed和@ObjectLink装饰器解决此问题,以确保组件间的高效数据同步。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一、使用场景

当对象内引用对象,改变内部对象属性的时候其他状态管理如@State、@Provide、@Consume等是无法触发更新的。同样,在数组内如果有对象,改变对象的属性也是无法更新的。在这种情况下就可以采用@Observed和@ObjectLink装饰器了。

二、使用方式

在我们需要监控的对象组件使用@Observed装饰,将需要变更的地方抽取出来作为一个新的组件,新组件内部引用对象使用@ObjectLink装饰。

例如, 在案例代码 这个基础上,我们需要在点击任务完成的时候,修改任务列表卡片的文字颜色,并中划线。这就涉及到了修改任务数组中Task对象的属性,需要更新UI。就需要@Observed和@ObjectLink结合使用。

 show code (在案例代码基础上做处理)

//需要监控的组件使用@Observed装饰
@Observed
class Task {
  static  id: number = 1
  name:string = '任务名称'+Task.id++
  finished:boolean = false
}

抽离之前任务列表的item为一个新组件

@Component
struct TaskItem {
    
//接收父组件中传递的task,使用@ObjectLink装饰
  @ObjectLink task: Task
    
 //父组件中处理数据更新的方法
  handTasks: () => void

  build() {
    Row() {
      if (this.task.finished) {
        Text(this.task.name)
          .tasksuccessed()
      } else {
        Text(this.task.name)
      }

      Checkbox()
        .select(this.task.finished)
        .onChange(value => {
          this.task.finished = value
          console.log('任务状态' + value + '')
           this.handTasks()
        })
    }
    .card()
    .justifyContent(FlexAlign.SpaceBetween)
  }
}
//父组件中的任务列表
      List({ space: 10 }) {
        ForEach(this.tasks, (task: Task, index) => {
          ListItem() {
            //将父组件的渲染交给新组件
            TaskItem({task:task,handTasks:this.handleTaskNumber.bind(this)})
          }
          .swipeAction({ end: this.deleteButton(index) })
        }, item => '' + item.name)
      }

其中,需要注意的第一点,传递父组件的方法时,传递的是this.handleTaskNumber没有()。如果传递()就表示方法调用了。

需要注意第二点,我们在将父组件方法传递给子组件的时候。如果没有bind(this)处理,默认调用的父组件方法的this就是子组件。所以,我们需要bind(this)永远使用父组件来调用父组件的方法。

<think>好的,我现在需要帮助用户理解鸿蒙开发中的@Observed@ObjectLink装饰器的作用,并找出错误的描述选项。首先,我需要回顾用户提供的引用内容,分析每个引用的关键点,然后结合这些信息来判断哪个选项是错误的。 根据用户提供的引用内容: 引用[1]指出在子组件中给@ObjectLink装饰的变量赋值是不允许的,并给出了反例。这说明@ObjectLink变量在子组件中是只读的,不能直接修改。 引用[2]提到在子组件中尝试给@ObjectLink变量赋值,如this.testNum = new Info(47),这会导致错误,进一步验证了引用[1]的内容。 引用[3]说明@ObjectLink用于接收@Observed类的实例,并与父组件中的状态变量建立双向绑定。这里强调@Observed装饰的类或属性,以及双向绑定的特性。 引用[4]讨论了@Provide@Consume的命名唯一性,与@Observed@ObjectLink无关,可能用于干扰选项。 接下来,用户的问题是关于这两个装饰器的作用解析及判断错误描述。假设用户提供的选项中可能包含以下几点: 1. @ObjectLink变量在子组件中可以修改。 2. @Observed装饰类,而@ObjectLink用于子组件建立双向绑定。 3. @ObjectLink只能用于基本数据类型。 4. @ObjectLink需要与父组件的@State变量绑定。 根据引用内容: - 错误描述可能是选项1,因为引用[1][2]明确禁止在子组件中修改@ObjectLink变量。 - 选项2正确,引用[3]支持。 - 选项3错误,因为@ObjectLink用于对象实例,而非基本类型。 - 选项4可能正确,因为需要父组件的对应状态变量,但需确认。 但用户提供的选项中是否包含这些需要推断。假设选项中有类似选项1的说法,则错误。例如,若选项A说“@ObjectLink允许子组件直接赋值”,则A错误。 总结:错误描述是允许在子组件中给@ObjectLink变量赋值。</think>### @Observed@ObjectLink装饰器作用解析 #### 核心功能说明 1. **@Observed装饰器** 用于标记类或类的属性,当被装饰的类属性发生变化时,会触发UI更新。例如: ```typescript @Observed class UserInfo { name: string; age: number; } ``` 此时若`UserInfo`实例的`name`或`age`变化,会通知绑定了该对象的组件更新[^3]。 2. **@ObjectLink装饰器** 在**子组件**中使用,用于接收父组件传递的`@Observed`类实例,并与之建立**单向同步**(非双向绑定)。 ```typescript @Component struct ChildComponent { @ObjectLink user: UserInfo; // 仅接受父组件传递的@Observed实例 } ``` **关键限制**:子组件中禁止直接对`@ObjectLink`变量赋值(如`this.user = new UserInfo()`),否则会报错[^1][^2]。 #### 错误描述判断 以下为典型错误描述: **错误选项示例**: > "子组件中可以通过`@ObjectLink`直接修改对象引用,例如`this.user = new UserInfo()`" **错误原因**: 根据引用[1]引用[2],子组件中给`@ObjectLink`变量赋值属于明确禁止的操作,只能通过父组件更新数据源。 #### 对比其他装饰器 | 装饰器 | 作用场景 | 数据流向
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值