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
对象,
他通过劫持对象的 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