vue3中ref及reactive的说明

本文介绍了Vue3中响应式的核心机制,包括ref和reactive的使用方法,以及在实际应用中可能遇到的响应式失效问题及其解决方案。重点强调了ref的灵活性和reactive的适用场景,并给出了统一使用ref以避免问题的建议。
摘要由CSDN通过智能技术生成

目录

1.响应式说明

2.vue3的ref及reactive的使用

3.reactive响应式失效问题

4.总结


1.响应式说明

vue的响应式是vue框架中的核心概念之一,它是指当数据发生变化时,vue能够自动更新视图。vue2的响应式是基于Object.defineProperty进行实现的。

当你把一个普通的 JavaScript 对象传入 Vue 实例作为 data 选项,Vue 将遍历此对象所有的 property,并使用 Object.defineProperty 把这些 property 全部转为 getter/setterObject.defineProperty 是 ES5 中一个无法 shim 的特性,这也就是 Vue 不支持 IE8 以及更低版本浏览器的原因。

这些 getter/setter 对用户来说是不可见的,但是在内部它们让 Vue 能够追踪依赖,在 property 被访问和修改时通知变更。这里需要注意的是不同浏览器在控制台打印数据对象时对 getter/setter 的格式化并不同,所以建议安装 vue-devtools 来获取对检查数据更加友好的用户界面。

每个组件实例都对应一个 watcher 实例,它会在组件渲染的过程中把“接触”过的数据 property 记录为依赖。之后当依赖项的 setter 触发时,会通知 watcher,从而使它关联的组件重新渲染。

vue3的响应式是通过ref及reactive来声明响应式状态。

当你在模板中使用了一个 ref,然后改变了这个 ref 的值时,Vue 会自动检测到这个变化,并且相应地更新 DOM。这是通过一个基于依赖追踪的响应式系统实现的。当一个组件首次渲染时,Vue 会追踪在渲染过程中使用的每一个 ref。然后,当一个 ref 被修改时,它会触发追踪它的组件的一次重新渲染。

在标准的 JavaScript 中,检测普通变量的访问或修改是行不通的。然而,我们可以通过 getter 和 setter 方法来拦截对象属性的 get 和 set 操作。

.value 属性给予了 Vue 一个机会来检测 ref 何时被访问或修改。在其内部,Vue 在它的 getter 中执行追踪,在它的 setter 中执行触发。

另一个 ref 的好处是,与普通变量不同,你可以将 ref 传递给函数,同时保留对最新值和响应式连接的访问。当将复杂的逻辑重构为可重用的代码时,这将非常有用。

当你修改了响应式状态时,DOM 会被自动更新。但是需要注意的是,DOM 更新不是同步的。Vue 会在“next tick”更新周期中缓冲所有状态的修改,以确保不管你进行了多少次状态修改,每个组件都只会被更新一次。

reactive() 将使对象本身具有响应性,响应式对象是 JavaScript 代理,其行为就和普通对象一样。不同的是,Vue 能够拦截对响应式对象所有属性的访问和修改,以便进行依赖追踪和触发更新。值得注意的是,reactive() 返回的是一个原始对象的 Proxy,它和原始对象是不相等的。只有代理对象是响应式的,更改原始对象不会触发更新。

2.vue3的ref及reactive的使用

①示例

<template>
    <div id="count1">
        count值: {{ count }}
    </div>
    <div>
        <div>
            姓名:{{ student.name }}
        </div>
        <div id="stu1">
            点赞数:{{ student.num }}
        </div>
    </div>
    <div>
        <button @click="btnClick()">按钮</button>
    </div>
</template>
<script setup>
// import {ref, reactive} from 'vue';
const ref = (target) => {
    let _value = target;
    return{
        get value() {
            return _value;
        },
        set value(newValue){
            _value = newValue;
            document.getElementById('count1').innerHTML = `count值: ${newValue}`;
        }
    }
}
const reactive = (target)=>{
return new Proxy(target,{
    get:(obj,key) =>{
        return obj[key];
    },
    set:(obj,key,newValue) =>{
        obj[key] = newValue;
        document.getElementById('stu1').innerHTML = `点赞数:${newValue}`;
        return true;
    }
})
}
const count = ref(0);
const student = reactive({
    name: '小李',
    num: 0
})

const btnClick = () => {
    count.value++;
    student.num++;
}
</script>

②说明

上面是一个简单的模拟ref及reactive的实现的示例。

ref说明:

ref可以声明任何类型的变量,包含基本类型及复杂类型。ref() 接收参数,并将其包裹在一个带有 .value 属性的 ref 对象中返回。

在js代码中需要通过ref对象的.value属性,然后再得到具体的内容进行赋值和使用,在template中可以直接使用,不需要进行.value操作。

ref的本质是在原有的传入数据的基础上,外层包了一层对象,包成了复杂类型,包裹成复杂类型后,再借助reactive实现响应式。

reactive说明:

reactive() 返回的是一个原始对象的 Proxy,只有代理对象是响应式的,更改原始对象不会触发更新。使用 Vue 的响应式系统的最佳实践是 仅使用你声明对象的代理版本

有限的值类型:它只能用于对象类型 (对象、数组和如 MapSet 这样的集合类型)。它不能持有如 stringnumberboolean 这样的原始类型

不能替换整个对象:由于 Vue 的响应式跟踪是通过属性访问实现的,因此我们必须始终保持对响应式对象的相同引用。这意味着我们不能轻易地“替换”响应式对象。

对解构操作不友好:当我们将响应式对象的原始类型属性解构为本地变量时,或者将该属性传递给函数时,我们将丢失响应性连接。

由于这些限制,我们建议使用 ref() 作为声明响应式状态的主要 API。

3.reactive响应式失效问题

1.reactive声明对象没有失效

<template>
    <div id="count1">
        count值: {{ count }}
    </div>
    <div>
        <div>
            姓名:{{ student.name }}
        </div>
        <div id="stu1">
            点赞数:{{ student.num }}
        </div>
    </div>
    <div>
        <button @click="btnClick()">按钮</button>
    </div>
</template>
<script setup>
import { ref, reactive } from 'vue';
const count = ref(0);
let student = reactive({
    name: '小李',
    num: 0
})

const btnClick = () => {
    count.value++;
    student = reactive({
        name: '小孙',
        num: 1
    })
}
</script>

点击按钮前:

点击按钮后:

可以看到值已经发生变化。

2.reactive声明数组响应式失效。

<template>
<div>
    {{ list }}
</div>
<div>
    <button  @click="btnClick()">按钮</button>
</div>

</template>

<script setup>

import {reactive} from 'vue';

let list = reactive([
    '123','234','345'
])
const btnClick = () =>{
    list = ['258','369','147'];
    console.log(list);
}
</script>

按钮点击前:

按钮点击后:

点击按钮后,list的值如控制台已经发生了变化,但是视图已经渲染,但是内容没有更新。此种方式不会触发视图更新,因为地址指向变成了一个新的地址。

解决方法:

①使用ref进行定义。将对象的.value设定为新的数组。

②将数组用对象的方式进行包裹,给对象中的属性进行赋值,在template中使用对象中的属性进行显示。如下:

<template>
<div>
    {{ list.arr }}
</div>
<div>
    <button  @click="btnClick()">按钮</button>
</div>

</template>

<script setup>

import {ref,reactive} from 'vue';

let list = reactive({
    arr:[
    '123','234','345'
]
})
const btnClick = () =>{
    list.arr = ['258','369','147'];
    console.log(list);
}
</script>

③采用push方式进行更新,注意要使用扩展运算符,否则会在数组中嵌套一个数据。

<template>
    <div>
        {{ list }}
    </div>
    <div>
        <button @click="btnClick()">按钮</button>
    </div>
</template>

<script setup>

import { ref, reactive } from 'vue';

let list = reactive([
    '123', '234', '345'
]
)
const btnClick = () => {
    list.push(...['258', '369', '147']);
    console.log(list);
}
</script>

 在使用reactive时需要使用const方式进行定义。通过const定义只能修改对象中属性,而不能修改整个对象。

4.总结

ref和reactive都是通过函数调用的方式生成响应式数据,从而达到画面自动更新的目的。ref可以处理任何类型,而reactive只能处理复杂类型。在开发中,推荐统一使用ref。这样既统一编码规范,又避免reactive的失效问题。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值