Vue3 ref构建响应式变量失效问题

在Vue3应用中遇到使用ref创建的响应式变量无法正常响应变化的问题。问题出在通过函数更新ref的值时未正确引用。通过修正引用方式,即从`this.myRef.value = ...`更改为`this.myRef.value = ...`,解决了响应式失效的状况。

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

问题描述

在Vue3中使用ref声明响应式变量,同时用函数对值进行变化,但是无法响应式改变值

<template>
  <p>{{userName}}</p>
  <button @click='change()'>change</button>
</template>

<script>
  //引入定义响应式数据的函数
  import {reactive} from 'vue';
  import {ref} from "@vue/reactivity"; //!!!!!注意,这里有个坑,ref必须是引用自vue,而非@vue/reactivity
  export default {
  name: 'App',
  //为Vue3的新特性提供统一入口,代码都会在这个函数中添加
  //在beforecreated之前进行,因此无法访问this,亦即无法访问data和method
  setup(){
    //定义响应式数据:数据变化,模板中渲染会自动刷新
    // const obj=reactive({
    //   userName:'jack',
    // });
    //只定义一个变量,可以使用ref将变量定义为响应式
    let userName=ref('jack')
    console.log(userName);
    const change=()=> {
      userName.value='rose'     //注意修改的是ref对象的value属性,但是在template中使用的时候不需要再加value
      console.log(userName);
    }

    return {userName,change}
  },
}
</script>


解决方案:

不知道为什么,当引用为

	import {ref} from "@vue/reactivity"

时,就会出现不响应的情况,但是只需要改为

	import {ref} from "vue"

就能成功解决问题了

### 解决 Vue 3 中 `v-for` 和 `defineModel` 失去响应式问题Vue 3 的组合式 API 下,当使用 `v-for` 渲染列表并结合 `defineModel` 定义模型时,可能会遇到组件失去响应式的现象。这通常是因为数据绑定的方式正确或者状态管理存在问题。 为了确保 `v-for` 结合 `defineModel` 正常工作,可以遵循以下方法: #### 使用 `ref` 或 `reactive` 对于简单的属性,推荐使用 `ref` 来创建响应式变量;而对于对象或数组,则应采用 `reactive` 函数来定义复杂的状态结构[^1]。 ```javascript import { ref, reactive } from &#39;vue&#39;; // 对于简单类型的响应式数据 const count = ref(0); // 对于复杂类型的响应式数据 const state = reactive({ items: [] }); ``` #### 确保正确更新状态 在修改由 `reactive` 创建的对象内部的数据时,应当通过该对象本身来进行操作而是直接替换整个对象实例,这样才能保持其子属性的响应特性。 ```javascript state.items.push(newItem); // 推荐做法 // 避免这样做: // state = {...state, items: [...items, newItem]}; ``` #### 组件间通信方式的选择 如果父级向子级传递的是一个基本类型值而非引用类型(如字符串、数字),那么即使是在子组件里改变了这个 prop 值也会影响到原始数据源。此时应该考虑利用事件机制通知父组件改变相应状态,或者是借助 Vuex 这样的全局状态管理模式来处理跨层级之间的交互逻辑。 #### 示例代码展示如何修复丢失反应性的错误 下面是一个具体的例子展示了怎样修正由于当使用引起的响应失效问题: ```html <template> <ul> <!-- 错误示范 --> <!-- <li v-for="(item, index) in nonReactiveItems" :key="index">{{ item }}</li> --> <!-- 正确示范 --> <li v-for="(item, index) in reactiveItems" :key="item.id"> {{ item.name }} </li> </ul> </template> <script setup> import { reactive } from &#39;vue&#39;; // 要这样写,因为普通的 JavaScript 数组响应式的 // const nonReactiveItems = [{ id: 1, name: &#39;Apple&#39; }, { id: 2, name: &#39;Banana&#39; }]; // 应该像这样声明响应式对象 const reactiveItems = reactive([ { id: 1, name: &#39;Apple&#39; }, { id: 2, name: &#39;Banana&#39; } ]); </script> ```
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

周飞飞丶

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值