• Vue 3 组合式 API 指南:响应式状态管理与跨组件通信


    引言

    随着 Vue 3 的发布,Vue 引入了一个新的编程范式——组合式 API(Composition API)。这一新特性旨在解决 Options API 在处理复杂逻辑时的一些局限性,并提供了一种更灵活、更可重用的方式来组织和重用代码

    组合式 API 基础

    setup 函数的概念和作用

    在 Vue 组合式 API 中,setup 函数是组件的入口点,它在组件实例被创建时调用,但在任何生命周期钩子之前执行。setup 函数的主要作用是初始化响应式状态、计算属性、方法和侦听器,并将它们暴露给模板和组件的其他部分。

    1. import { ref, reactive, computed, watch } from 'vue';
    2. export default {
    3. setup() {
    4. // 初始化响应式状态和功能
    5. return {
    6. // 暴露给模板和组件其他部分的响应式状态和功能
    7. };
    8. }
    9. };
    响应式引用 ref 和响应式对象 reactive 的使用方法

    在 Vue 3 中,ref 和 reactive 是创建响应式状态的两个主要工具。

    • ref 用于创建单个响应式值,它返回一个响应式的引用对象。当引用的值是基本数据类型时,Vue 会自动将其包装在对象中。
    1. import { ref } from 'vue';
    2. export default {
    3. setup() {
    4. const count = ref(0);
    5. // 通过 .value 访问和修改响应式值
    6. console.log(count.value); // 输出 0
    7. count.value++;
    8. console.log(count.value); // 输出 1
    9. return {
    10. count
    11. };
    12. }
    13. };
    • reactive 用于创建一个响应式对象。它返回一个响应式的代理对象,该对象的属性访问和修改都是响应式的。(实际上,ref底层就是reactive实现的,实际开发中一般ref用的多)
    1. import { reactive } from 'vue';
    2. export default {
    3. setup() {
    4. const state = reactive({ count: 0 });
    5. // 直接访问和修改响应式对象的属性
    6. console.log(state.count); // 输出 0
    7. state.count++;
    8. console.log(state.count); // 输出 1
    9. return {
    10. state
    11. };
    12. }
    13. };
    使用 computed 和 watch 创建计算属性和侦听器

    computed 和 watch 是 Vue 组合式 API 中用于处理计算属性和侦听器的工具。

    • computed 用于创建计算属性,它接受一个 getter 函数,并返回一个只读的响应式引用。
    1. import { ref, computed } from 'vue';
    2. export default {
    3. setup() {
    4. const count = ref(0);
    5. const doubleCount = computed(() => count.value * 2);
    6. // 访问计算属性
    7. console.log(doubleCount.value); // 输出 0
    8. return {
    9. count,
    10. doubleCount
    11. };
    12. }
    13. };
    • watch 用于创建侦听器,它监视一个或多个响应式引用的变化,并在变化时执行一个回调函数。
    1. import { ref, watch } from 'vue';
    2. export default {
    3. setup() {
    4. const count = ref(0);
    5. watch(count, (newValue, oldValue) => {
    6. console.log(`Count changed from ${oldValue} to ${newValue}`);
    7. });
    8. return {
    9. count
    10. };
    11. }
    12. };

    provide 和 inject

    Vue.js 中,provide 和 inject 是两个用于实现跨组件依赖注入的 API。它们允许一个祖先组件定义可供其所有子孙组件使用的数据或方法,而无需通过逐层传递 props 或使用事件发射器。

    provide

    provide 函数允许你在组件内部提供一个值,这个值可以被其所有子孙组件注入。provide 可以在组件的 setup 函数中使用,也可以在组件的 methodscomputed 或 watch 中使用。

    1. import { provide } from 'vue';
    2. export default {
    3. setup() {
    4. // 提供一个响应式对象
    5. const state = reactive({ count: 0 });
    6. // 提供一个方法
    7. const increment = () => {
    8. state.count++;
    9. };
    10. // 使用 provide 函数提供数据
    11. provide('state', state);
    12. provide('increment', increment);
    13. // ...
    14. }
    15. };

    在上面的例子中,state 和 increment 方法被提供给所有子孙组件。

    inject

    inject 函数用于在子孙组件中注入由祖先组件提供的数据或方法。inject 可以在组件的 setup 函数中使用,也可以在组件的 methodscomputed 或 watch 中使用。

    1. import { inject } from 'vue';
    2. export default {
    3. setup() {
    4. // 注入 'state''increment'
    5. const state = inject('state');
    6. const increment = inject('increment');
    7. // 使用注入的数据和方法
    8. console.log(state.count); // 输出 0
    9. increment(); // 增加计数
    10. // ...
    11. }
    12. };

    在上面的例子中,state 和 increment 被注入到子孙组件中,并可以像本地数据一样使用。

    注意事项

    • provide 和 inject 是成对使用的,只有当祖先组件提供了某个值,子孙组件才能注入这个值。
    • provide 和 inject 提供的值是响应式的,这意味着如果提供的值发生变化,所有注入了这个值的子孙组件都会自动更新。
    • provide 和 inject 可以在组件的任何生命周期钩子中使用,但通常在 setup 函数中使用,以确保在组件初始化时提供和注入值。

    通过使用 provide 和 inject,开发者可以更灵活地管理组件之间的依赖关系,特别是在大型应用中,这有助于减少组件之间的耦合度,提高代码的可维护性。

     子孙组件能改动inject注入的值吗

    孙组件不能直接修改通过 inject 注入的响应式数据。这是为了保持数据流的清晰和组件之间的独立性,避免潜在的副作用和难以追踪的错误。

    然而,如果你确实需要在子孙组件中修改注入的值,你可以通过以下方法实现:

    1.使用 ref 或 reactive 提供响应式数据
    如果你使用 ref 或 reactive 提供响应式数据,那么子孙组件可以通过 .value 属性来修改这个值。

    1. // 祖先组件
    2. import { provide, reactive } from 'vue';
    3. export default {
    4. setup() {
    5. const state = reactive({ count: 0 });
    6. provide('state', state);
    7. // ...
    8. }
    9. };
    10. // 子孙组件
    11. import { inject } from 'vue';
    12. export default {
    13. setup() {
    14. const state = inject('state');
    15. // 修改注入的响应式数据
    16. state.count++;
    17. // ...
    18. }
    19. };

    2.提供一个方法来修改数据
    你可以在祖先组件中提供一个方法,该方法可以修改响应式数据。然后,这个方法可以被子孙组件注入并调用。

    1. // 祖先组件
    2. import { provide } from 'vue';
    3. export default {
    4. setup() {
    5. const state = reactive({ count: 0 });
    6. const increment = () => {
    7. state.count++;
    8. };
    9. provide('state', state);
    10. provide('increment', increment);
    11. // ...
    12. }
    13. };
    14. // 子孙组件
    15. import { inject } from 'vue';
    16. export default {
    17. setup() {
    18. const increment = inject('increment');
    19. // 调用注入的方法来修改数据
    20. increment();
    21. // ...
    22. }
    23. };

    3.使用 readonly 创建只读引用
    如果你不希望子孙组件修改注入的值,可以使用 readonly 创建一个只读引用。

    1. // 祖先组件
    2. import { provide, reactive, readonly } from 'vue';
    3. export default {
    4. setup() {
    5. const state = reactive({ count: 0 });
    6. provide('state', readonly(state));
    7. // ...
    8. }
    9. };
    10. // 子孙组件
    11. import { inject } from 'vue';
    12. export default {
    13. setup() {
    14. const state = inject('state');
    15. // 尝试修改注入的值将会失败
    16. // state.count = 1; // 这将不会工作
    17. // ...
    18. }
    19. };

    总结

    Vue 3 的组合式 API 通过 setup 函数提供响应式状态管理,支持使用 refreactive 创建响应式数据,computedwatch 处理计算属性和侦听器。同时,provideinject 允许跨组件进行依赖注入,实现数据和方法的共享。这些特性提升了代码的组织性、可重用性以及组件间的解耦。

    希望这篇文章能帮助到你

  • 相关阅读:
    【C++面向对象】7. 类的静态成员
    D. Count GCD(数论/gcd/素数筛/容斥)
    ES6(ECMASript 相关介绍,ECMASript 6 新特性)
    opencv交互式调整视觉算法参数(一)-图像阈值参数
    造轮子之ORM集成
    1333:【例2-2】Blah数集
    【源码+文档+调试讲解】基于vue的线上点餐系统
    有哪些美股量化接口?
    四大特性模块(module)
    MySQL 8.0性能优化实战培训
  • 原文地址:https://blog.csdn.net/a3098448071/article/details/140408524