当前位置:   article > 正文

Vue3超详细的ref()用法,看这一篇就够了_vue3 ref

vue3 ref

ref( ) 接受一个内部值,返回一个ref 对象,这个对象是响应式可更改的,且只有一个指向其内部值的属性 .value

ref() 将传入参数的值包装为一个带 .value 属性的 ref 对象。

1、ref 对象是可更改的,即可以为 .value 赋予新的值

举例:

  1. const a = ref(1);
  2. // 为 a.value 赋予新的值
  3. a.value = 2;
  4. console.log("a--->", a);
  5. console.log("a.value--->", a.value);

查看打印结果:

2、ref 对象是响应式的,即所有对 .value 的操作都将被追踪,并且写操作会触发与之相关的副作用。

ref()方法允许创建可以使用任何值类型的响应式 ref

ref 的 .value 属性也是响应式的。

当ref的值为对象类型时,会用 reactive() 自动转换它的 .value。

举例:一个包含对象类型值的 ref 可以响应式地替换整个对象

  1. const b = ref({ name: 'vue3' });
  2. // 响应式替换
  3. b.value = { name: 'vite' };
  4. console.log("b--->", b);
  5. console.log("b.value--->", b.value);

查看打印结果:

ref 被传递给函数或是从一般对象上被解构时,不会丢失响应性:

  1. const obj = {
  2. foo: ref(0),
  3. bar: ref(1)
  4. }
  5. // 该函数接收一个 ref
  6. // 需要通过 .value 取值
  7. // 但它会保持响应性
  8. callSomeFunction(obj.foo);
  9. // 仍然是响应式的
  10. const { foo, bar } = obj;

总结:ref() 让我们能创造一种对任意值的 “引用”,并能够在不丢失响应性的前提下传递这些引用。这个功能很重要,因为它经常用于将逻辑提取到组合函数中。

3、ref 在模板中的解包

当 ref 在模板中作为顶层属性被访问时,它们会被自动“解包”,所以不需要使用 .value。

  1. <script setup>
  2. import { ref } from 'vue';
  3. const a = ref(1);
  4. </script>
  5. <template>
  6.   <!-- 无需 .value -->
  7. <div>a:{{ a }}</div>
  8. </template>

⚠️请注意,仅当 ref 是模板渲染上下文的顶层属性时才适用自动“解包”。

  1. <script setup>
  2. import { ref } from 'vue';
  3. const obj = {
  4. count: ref(1)
  5. }
  6. </script>
  7. <template>
  8. <div>{{ obj.count + 1 }}</div>
  9. </template>

渲染的结果是 [object Object]1,因为 object.count 是一个 ref 对象

可以通过将 count 改成顶层属性来解决这个问题:

  1. <script setup>
  2. import { ref } from 'vue';
  3. const obj = {
  4. count: ref(1)
  5. }
  6. // 将 count 改成顶层属性
  7. const { count } = obj;
  8. </script>
  9. <template>
  10. <div>{{ count + 1 }}</div>
  11. </template>

渲染结果是 2

⚠️如果一个 ref 是文本插值计算的最终值,它也将被解包

  1. <script setup>
  2. import { ref } from 'vue';
  3. const obj = {
  4. count: ref(1)
  5. }
  6. const { count } = obj;
  7. </script>
  8. <template>
  9. <div>{{ count + 1 }}</div>
  10. <div class="count">{{ obj.count }}</div>
  11. </template>

<div class="count">{{ obj.count }}</div>的渲染结果为 1

这只是文本插值的一个方便功能,相当于 {{ object.foo.value }}

4、ref 在响应式对象中的解包

当一个 ref 被嵌套在一个响应式对象中,作为属性被访问或更改时,它会自动解包,因此会表现得和一般的属性一样:

  1. import { ref, reactive } from 'vue';
  2. const a = ref(0);
  3. const obj = reactive({
  4. a
  5. })
  6. console.log("obj.a--->", obj.a);
  7. obj.a = 2;
  8. console.log("a.value--->", a.value);

查看打印结果:

如果将一个新的 ref 赋值给一个关联了已有 ref 的属性,那么它会替换掉旧的 ref:

  1. import { ref, reactive } from 'vue';
  2. const a = ref(0);
  3. const other = ref(1);
  4. const obj = reactive({
  5. a
  6. })
  7. // 将一个新的 ref 赋值给一个关联了已有 ref 的属性
  8. obj.a = other;
  9. console.log("obj.a--->", obj.a);
  10. // 原始 ref 现在已经和 obj.a 失去联系
  11. console.log("a.value--->", a.value);

查看打印结果:

只有当嵌套在一个深层响应式对象内时,才会发生 ref 解包。当其作为浅层响应式对象的属性被访问时不会解包。

5、ref在数组和集合类型的解包

跟响应式对象不同,当 ref 作为响应式数组或像 Map 这种原生集合类型的元素被访问时,不会进行解包。

  1. import { ref, reactive } from 'vue';
  2. const books = reactive([ref('Vue 3 Guide')]);
  3. // 这里需要 .value
  4. console.log(books[0].value);
  5. const map = reactive(new Map([['count', ref(0)]]));
  6. // 这里需要 .value
  7. console.log(map.get('count').value);

6、ts为 ref() 标注类型

ref 会根据初始化时的值推导其类型:

  1. import { ref } from 'vue'
  2. // 推导出的类型:Ref<number>
  3. const year = ref(2020)
  4. // => TS Error: Type 'string' is not assignable to type 'number'.
  5. year.value = '2020'

有时我们可能想为 ref 内的值指定一个更复杂的类型,可以通过使用 Ref 这个类型

  1. import { ref } from 'vue'
  2. import type { Ref } from 'vue'
  3. const year: Ref<string | number> = ref('2020')
  4. year.value = 2020 // 成功!

或者,在调用 ref() 时传入一个泛型参数,来覆盖默认的推导行为:

  1. // 得到的类型:Ref<string | number>
  2. const year = ref<string | number>('2020')
  3. year.value = 2020 // 成功!

如果你指定了一个泛型参数但没有给出初始值,那么最后得到的就将是一个包含 undefined 的联合类型:

  1. // 推导得到的类型:Ref<number | undefined>
  2. const n = ref<number>()
声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/神奇cpp/article/detail/910988
推荐阅读
相关标签
  

闽ICP备14008679号