外观
Vue3-ref&reactive
1457字约5分钟
2024-06-13
Vue3-ref&reactive
ref 与 reactive
ref 基础类型响应式
- 作用: 定义响应式变量。
- 语法:
let xxx = ref(初始值)
。 - 返回值: 一个
RefImpl
的实例对象,简称ref对象
或ref
,ref
对象的value
属性是响应式的。
提示
JS
中操作数据需要:xxx.value
,但模板中不需要.value
,直接使用即可。- 对于
let total = ref(1)
来说,total
不是响应式的,total.value
是响应式的。
<template>
<div class="hello">
{{ msg }}
<button @click="clickChange">Click</button>
{{total}}
</div>
</template>
<script setup lang="ts" name="demo">
import {ref} from 'vue'
let msg = "Hello Vue3";
let total = ref(1);
console.log(total)
console.log(total.value)
function clickChange(){
total.value++
}
</script>
打印输出
RefImpl {__v_isShallow: false, dep: undefined, __v_isRef: true, _rawValue: 1, _value: 1}
1
点击按钮是 页面也会改变 total
数据变化
reactive 对象类型响应式
- 作用: 定义一个响应式对象(基本类型不要用它,要用
ref
,否则报错) - 语法:
let 响应式对象= reactive(源对象)
。 - 返回值: 一个
Proxy
的实例对象,简称:响应式对象。 - 注意点:
reactive
定义的响应式数据是“深层次”的。
<template>
<div class="hello">
{{ data.name }}{{data.msg}}
<button @click="clickChange">Click</button>
{{ data.id }}
<ul>
<li v-for="item in list" :key="item.id">{{item.name}}:{{item.msg}}</li>
</ul>
</div>
</template>
<script setup lang="ts" name="demo">
import { reactive } from "vue";
let data = reactive({ id: 1, name: "yevin", msg: "hello, everyone!" });
let list = reactive([
{ id: 1, name: "wenbo", msg: "ni hao" },
{ id: 2, name: "yiran", msg: "hello" },
]);
function clickChange() {
data.id++;
list[0].msg += ' yevin '
console.log(data);
console.log(list);
}
</script>
点击按钮,页面id数据更新,控制台打印如下:
Proxy(Object) {id: 2, name: 'yevin', msg: 'hello, everyone!'}
Proxy(Array) {0: {…}, 1: {…}}
Proxy(Object) {id: 3, name: 'yevin', msg: 'hello, everyone!'}
Proxy(Array) {0: {…}, 1: {…}}
ref 对象类型响应式
- 其实
ref
接收的数据可以是:基本类型、对象类型。 - 若
ref
接收的是对象类型,内部其实也是调用了reactive
函数。
提示
ref
定义的数据,需要通过.value
获取
<template>
<div class="hello">
{{ data.name }}{{data.msg}}
<button @click="clickChange">Click</button>
{{ data.id }}
<ul>
<li v-for="item in list" :key="item.id">{{item.name}}:{{item.msg}}</li>
</ul>
</div>
</template>
<script setup lang="ts" name="demo">
import { ref } from "vue";
let data = ref({ id: 1, name: "yevin", msg: "hello, everyone!" });
let list = ref([
{ id: 1, name: "wenbo", msg: "ni hao" },
{ id: 2, name: "yiran", msg: "hello" },
]);
function clickChange() {
data.value.id++;
list.value[0].msg += ' yevin '
console.log(data);
console.log(list);
}
</script>
点击按钮,页面id数据更新,控制台打印如下:
RefImpl {__v_isShallow: false, dep: Map(1), __v_isRef: true, _rawValue: {…}, _value: Proxy(Object)}
RefImpl {__v_isShallow: false, dep: Map(1), __v_isRef: true, _rawValue: Array(2), _value: Proxy(Array)}
RefImpl {__v_isShallow: false, dep: Map(1), __v_isRef: true, _rawValue: {…}, _value: Proxy(Object)}
RefImpl {__v_isShallow: false, dep: Map(1), __v_isRef: true, _rawValue: Array(2), _value: Proxy(Array)}
ref 和 reactive 对比
- 基础:
ref
用来定义:基本类型数据、对象类型数据;
reactive
用来定义:对象类型数据。
- 区别:
ref
创建的变量必须使用.value
(可以使用volar
插件自动添加.value
)。
reactive
重新分配一个新对象,会失去响应式(可以使用Object.assign
去整体替换)。
let data = reactive({ id: 1, name: "yevin", msg: "hello, everyone!" })
// 错误示例
// data = {id: 100, name: "wenbo", msg: "wenbo" } // 失去响应式
// data = reactive({id: 100, name: "wenbo", msg: "wenbo" }) // 不行 XXXX
// 正确示例
Object.assign(data,{id: 100, name: "wenbo", msg: "wenbo" })
- 使用原则:
- 若需要一个基本类型的响应式数据,必须使用
ref
。
- 若需要一个响应式对象,层级不深,
ref
、reactive
都可以。
- 若需要一个响应式对象,且层级较深,推荐使用
reactive
。
ref
和 .value
- 当我们直接使用ref定义数据时,获取数据需要
.value
let num = ref(1)
let obj = ref({name:'yuwb'})
console.log(num.value)
console.log(obj.value)
- 当ref定义的数据,外面有ractive包裹时,不需要使用
.value
。因为ractive定义的响应式对象里面的属性在被访问的时候自动接解包。
const data = reactive({
name:"yuwb",
age:ref(18)
})
console.log(data.age)
toRef 与 toRefs
- 作用:将一个响应式对象中的每一个属性,转换为
ref
对象。 - 备注:
toRefs
与toRef
功能一致,但toRefs
可以批量转换。
使用场景
<template>
<div class="hello">
{{ name }}{{ data.msg }}
<button @click="clickChange">Click</button>
{{ id}}
</div>
</template>
<script setup lang="ts" name="demo">
import { reactive } from "vue";
let data = reactive({ id: 1, name: "yevin", msg: "hello, everyone!" });
let { id, name } = data;
function clickChange() {
console.log(id,name) // 1,yevin
id = 100;
name = "wenbo";
console.log(id,name) // 100,wenbo
}
</script>
在上述代码中,解构一个响应式的数据后,更改数据,可以看到数据更改了但是页面并没有更新。
原因:
解构代码中相当于 单独定义了 id 和 name数据,并且没有绑定响应式。
let { id, name } = data;
// 上述代码相当于 单独定义了 id 和 name数据,并且没有绑定响应式。
let id = data.id
let name = data.name
使用toRefs
优化代码:
<template>
<div class="hello">
{{ name }}{{ data.msg }}
<button @click="clickChange">Click</button>
{{ id}}
</div>
</template>
<script setup lang="ts" name="demo">
import { reactive ,toRefs} from "vue";
let data = reactive({ id: 1, name: "yevin", msg: "hello, everyone!" });
let obj = toRefs(data);
console.log(obj)
let { id, name } = obj
function clickChange() {
id.value = 100;
name.value = "wenbo";
console.log(id,name)
console.log(id.value,name.value)
console.log(data)
}
</script>
点击按钮输出如下:
{id: ObjectRefImpl, name: ObjectRefImpl, msg: ObjectRefImpl}
ObjectRefImpl {_object: Proxy(Object), _key: 'id', _defaultValue: undefined, __v_isRef: true}
ObjectRefImpl {_object: Proxy(Object), _key: 'name', _defaultValue: undefined, __v_isRef: true}
100 'wenbo'
Proxy(Object) {id: 100, name: 'wenbo', msg: 'hello, everyone!'}
通过上述代码可以看到:toRefs
将一个响应式对象中的每一个属性,转换为ref
对象,因此解构获得是对应的ref
对象,都有绑定响应式。
所以,改变通过解构获取的值之后,页面会同步更新,原数据也会同步更新。
提示
注意:通过toRefs
和toRef
操作之后,获取值或者更改数据需要使用.value
。
toRef与toRefs
使用reactive
定义对象类型响应
<script setup lang="ts" name="demo">
import { reactive ,toRefs,toRef} from "vue";
let data = reactive({ id: 1, name: "yevin", msg: "hello, everyone!" });
// toRefs使用
let { id, name } = toRefs(data)
// toRef使用
let msgData = toRef(data,'msg')
</script>
使用ref
定义对象类型响应
<script setup lang="ts" name="demo">
import { ref, toRefs, toRef } from "vue";
let data = ref({ id: 1, name: "yevin", msg: "hello, everyone!" });
// toRefs使用
let { id, name } = toRefs(data.value);
// toRef使用
let msgData = toRef(data.value, "msg");
</script>