reactive - Vue3 composition(组合式)API

reactive()--定义响应式数据

reactive()接受一个对象,包括json数据和数组都可以,返回一个对象的响应式代理(Proxy对象)。

<script src="https://unpkg.com/vue@next"></script>
<div id="Application">

</div>
<script>
    const App = Vue.createApp({
        //进行组件数据初始化
        setup() {
            //数据
            //基本类型使用ref,引用类型使用reactive
            //增删改查都可以操作。
            //内部是通过Proxy代理的形式
            let myData = Vue.reactive({
                value: 0
            })
            //按钮的单击方法
            function click() {
                myData.value += 1
                console.log(myData.value)
            }
            //数据返回
            return {
                myData,
                click
            }
        },
        //模板中可以直接使用setup方法中定义的数据和函数
        template: `
        <h1>测试数据:{{ myData.value }}</h1>
        <button @click="click">点击</button>
        `
    })
    App.mount("#Application")
</script>

您可以打开浏览器控制台,观察控制台的打印数据

  • eactive 是 Vue3 中提供的实现响应式数据的方法。
  • 在 Vue2 中响应式数据是通过 defineProperty 来实现的,
  • 在 Vue3 中响应式数据是通过 ES6 的 Proxy来实现的。
  • reactive 参数必须是对象 (json / arr)
  • 如果给 reactive 传递了其它对象
  • 默认情况下,修改对象无法实现界面的数据绑定更新。
  • 如果需要更新,需要进行重新赋值。(即不允许直接操作数据,需要放个新的数据来替代原数据)

在 reactive 使用基本类型参数

基本类型(数字、字符串、布尔值)在 reactive 中无法被创建成 proxy 对象,也就无法实现监听。

<script src="https://unpkg.com/vue@next"></script>
<div id="Application">
</div>

<script>
    const App = Vue.createApp({
        setup() {
            let msg = Vue.reactive(0)
            function click() {
                console.log(msg);
                msg++;
            }
            return {
                msg,
                click
            };
        },
        template: `
        <h1>测试数据:{{ msg }}</h1>
        <button @click="click">点击</button>
        `
    })
    App.mount("#Application")

</script>

点击 按钮 ,我们期望的结果是数字从 0 变成 1,然而实际上界面上的数字并没有发生任何改变。

查看控制台,它的输出是这样的(我点了 3 次)(您可以尝试F12打开控制台实时观察效果)

1
2
3

出现提示

value cannot be made reactive: 0

而输出的值确实发生了变化,只不过这种变化并没有反馈到界面上,也就是说并没有实现双向数据绑定。当然,如果是 ref 的话,就不存在这样的问题。

而如果要使用 reactive ,我们需要将参数从 基本类型 转化为 对象。

<script src="https://unpkg.com/vue@next"></script>
<div id="Application">
</div>

<script>
    const App = Vue.createApp({
        setup() {
            let msg = Vue.reactive({ value: 0 })
            function click() {
                console.log(msg);
                msg.value++;
            }
            return {
                msg,
                click
            };
        },
        template: `
        <h1>测试数据:{{ msg.value }}</h1>
        <button @click="click">点击</button>
        `
    })
    App.mount("#Application")

</script>

将参数替换成了对象 {num: 0},此时,点击按钮界面就会产生改变(我点了 3 次)。

控制台打印消息(您可以在本页点击按钮,观察控制台变化)

Proxy {value: 0}
[[Handler]]: Object
[[Target]]: Object
[[IsRevoked]]: false
 Proxy {value: 1}
Proxy {value: 2}

可以看到,msg 成功被创建成了 proxy 对象,

reactive - Vue3 composition(组合式)API-Npcink
reactive - Vue3 composition(组合式)API-Npcink

Proxy - Javascript中的代理器

他通过劫持对象的 get 和 set 方法实现了对象的双向数据绑定。

对象内部变化

深层的、对象内部的变化也能被察觉到(注意下面代码中的 inner )

<script src="https://unpkg.com/vue@next"></script>

<div id="Application">
</div>

<script>
    const App = Vue.createApp({
        setup() {
            let msg = Vue.reactive({
                value: {
                    inner: 0
                }
            })
            function click() {
                console.log(msg);
                msg.value.inner++;
            }
            return {
                msg,
                click
            };
        },
        template: `
        <h1>测试数据:{{ msg.value.inner }}</h1>
        <button @click="click">点击</button>
        `
    })
    App.mount("#Application")

</script>

可点击按钮,打开浏览器控制台观察变化

数组变化

<script src="https://unpkg.com/vue@next"></script>

<div id="Application">
</div>

<script>
    const App = Vue.createApp({
        setup() {
            let msg = Vue.reactive([1, 2, 3])
            function click() {
                console.log(msg);
                msg[0] += 1;
                msg[1] = 5;
            }
            return {
                msg,
                click
            };
        },
        template: `
        <h1>测试数据:{{ msg }}</h1>
        <button @click="click">点击</button>
        `
    })
    App.mount("#Application")

</script>

可点击按钮,打开浏览器控制台观察变化

重新赋值

在 reactive 使用 Date 参数

如果参数不是数组、对象,而是稍微奇怪一点的数据类型,例如说 Date,我们需要重新赋值

<script src="https://unpkg.com/vue@next"></script>

<div id="Application">
</div>

<script>
    const App = Vue.createApp({
        setup() {
            let msg = Vue.reactive({
                date: new Date()
            })
            function click() {
                console.log("第一次打印", msg);
                msg.date.setDate((msg.date.getDate() + 1));
                msg.date = new Date(msg.date);
                console.log("第二次打印", msg)

            }
            return {
                msg,
                click
            };
        },
        template: `
        <h1>测试数据:{{ msg.date }}</h1>
        <button @click="click">点击</button>
        `
    })
    App.mount("#Application")

</script>

这里我采用了拷贝的方案重新赋值了 msg.date,界面成功发生了变化(日期 + 1)。

响应式代理 vs. 原始对象

值得注意的是,reactive() 返回的是一个源对象的 Proxy,它和源对象是不相等的:

const raw = {}
const proxy = reactive(raw)
 
// 代理和原始对象不是全等的
console.log(proxy === raw) // false

只有代理是响应式的,更改原始的对象不会触发更新。因此,使用 Vue 的响应式系统的最佳实践是 仅使用代理作为状态。

为保证访问代理的一致性,对同一个对象调用 reactive() 会总是返回同样的代理,而对代理调用 reactive() 则会返回它自己:

// 在同一个对象上调用 reactive() 会返回相同的代理
console.log(reactive(raw) === proxy) // true
// 在一个代理上调用 reactive() 会返回它自己
console.log(reactive(proxy) === proxy) // true

这个规则对深层级的对象也适用。依靠深层响应性,响应式对象内的深层级对象依然是代理:

const proxy = reactive({})
const raw = {}
proxy.nested = raw
console.log(proxy.nested === raw) // false

参考文章

VUE模块

methods方法 - Vue3新手基础教程

2022-8-22 17:43:08

VUE模块

Vue3 composition(组合式)API详解

2022-9-1 17:09:57

⚠️
Npcink上的部份代码及教程来源于互联网,仅供网友学习交流,若您喜欢本文可附上原文链接随意转载。
无意侵害您的权益,请发送邮件至 1355471563#qq.com 或点击右侧 私信:Muze 反馈,我们将尽快处理。
0 条回复 A文章作者 M管理员
    暂无讨论,说说你的看法吧
个人中心
购物车
优惠劵
今日签到
有新私信 私信列表
搜索