揭开 Vue 3 中大量使用 ref 的隐藏危机

news/2024/9/20 20:40:51 标签: 前端, javascript, vue.js, ref

在 Vue 3 中,ref 是用来创建响应式的引用,它能够追踪和管理单一的变量或对象。当代码中大量使用 ref 时,虽然可以实现对各个状态或数据的精细控制,但也会带来一些问题和潜在影响。

1. 大量使用 ref 带来的问题

1、代码冗长与维护成本高

当一个组件中大量使用 ref,每个状态都需要单独的声明和赋值,会导致代码变得冗长、不易阅读。而且会频繁看到 ref(变量)和 变量.value 的形式,这使得代码的可读性降低,维护变得更加困难。

2、响应式系统性能负担

每个 ref 创建的响应式对象都要被 Vue 的响应式系统追踪和更新。当存在大量的 ref ,Vue 的响应式系统需要处理更多的状态变化,可能会对性能产生负面影响,特别是在大型应用或复杂组件中。

3、频繁 .value 操作

ref 包装的值需要通过 .value 来访问或更新,这在某些情况下容易引发不必要的重复代码,降低代码的简洁性。此外,使用时还需特别注意,什么时候该使用 .value,什么时候不使用。

4、数据管理混乱

当一个组件中有太多 ref,会导致管理状态变得复杂,难以明确哪个 ref 控制哪个部分的数据或逻辑,增加了组件的复杂度。

2. 改进方法

2.1 使用 reactive 替代 ref

如果需要管理多个相关的状态,建议使用 reactive 而不是多个 ref。reactive 可以将整个对象变成响应式的,这样就可以操作对象的属性来管理多个状态,而不需要单独创建多个 ref

举个 🌰

1、使用 ref

<template>
  <div class="container">
    <p>name:{{ name }}</p>
    <p>age:{{ age }}</p>
    <p>address:{{ address }}</p>
  </div>
</template>

<script>
import { ref } from 'vue';
export default {
  setup() {
    const name = ref('');
    const age = ref(0);
    const address = ref('');
    const createUser = () => {
      name.value = 'John';
      age.value = 20;
      address.value = 'New York';
    };
    createUser();
    return {
      name,
      age,
      address,
    };
  },
};
</script>

2、优化后使用 reactive 

<template>
  <div class="container">
    <p>name:{{ user.name }}</p>
    <p>age:{{ user.age }}</p>
    <p>address:{{ user.address }}</p>
  </div>
</template>

<script>
import { reactive } from 'vue';
export default {
  setup() {
    const user = reactive({
      name: '',
      age: 0,
      address: '',
    });
    const createUser = () => {
      user.name = 'John';
      user.age = 20;
      user.address = 'New York';
    };
    createUser();
    return {
      user,
    };
  },
};
</script>

展示都是一样的:

在这个例子中,使用 reactive 将 name、age 和 address 放在同一个对象中管理,减少了多个 ref,使代码更加简洁,且更容易维护。

2.2 使用 computed 来处理派生状态

如果某个状态是从其他状态推导出来的,那么可以使用 computed 来代替 ref,从而避免创建额外的响应式数据。

举个 🌰

ref="/tags/JAVASCRIPT.html" title=javascript>javascript">import { ref, computed } from 'vue';

export default {
  setup() {
    const firstName = ref('John');
    const lastName = ref('Doe');
    // 使用 computed 来合成新的状态,而不是通过 ref 手动管理
    const fullName = computed(() => `${firstName.value} ${lastName.value}`);
    return {
      firstName,
      lastName,
      fullName,
    };
  },
};
2.3 使用组合式函数 (Composables)

当状态逻辑非常复杂时,考虑将逻辑拆分成多个组合式函数(Composables)。这不仅有助于代码组织,还能在不同组件间共享相似的逻辑,而不是在每个组件中重复定义多个 ref

举个 🌰

<template>
  <div class="container">
    <p>name:{{ user.name }}</p>
    <p>age:{{ user.age }}</p>
    <p>address:{{ user.address }}</p>
  </div>
</template>

<script>
import { ref, reactive } from 'vue';
// 组合式函数:管理用户信息
function useUser() {
  const user = reactive({
    name: '',
    age: 0,
    address: '',
  });
  const updateUser = (newUser) => {
    user.name = newUser.name;
    user.age = newUser.age;
    user.address = newUser.address;
  };
  return {
    user,
    updateUser,
  };
}
export default {
  setup() {
    const { user, updateUser } = useUser();
    updateUser({
      name: 'Monica',
      age: 18,
      address: 'China',
    });
    return {
      user,
    };
  },
};
</script>

展示为:

这样可以将复杂的逻辑拆分到组合式函数中,在不同组件间复用状态和方法,减少了组件内部的状态管理复杂度。

3. 注意事项

1、ref vs reactive 区别

虽然 ref 和 reactive 都可以实现响应式状态管理,但 ref 适合处理单一原始值(如数字、字符串等),而 reactive 适合处理对象。为了避免不必要的复杂性,当有多个相关状态时优先使用 reactive。

2、.value` 的使用

ref 的 .value 访问方式在 Vue 3 是强制的(除非使用解构 toRefs 等)。频繁使用 .value 会显得繁琐,所以如果要管理多个属性,最好使用 reactive 来避免这些问题。

3、watch 和 ref 

当使用 watch 监听 ref 时,注意 .value 的引用变化。Vue 的 watch 默认是深度监听(deep),在处理复杂对象时可以通过配置 deep: false 来避免不必要的性能开销。

总结:

使用大量 ref 可能导致代码冗长、性能问题和维护难度的增加。为了解决这些问题,可以采用 reactive 管理多个状态,使用 computed 来处理派生状态,或者使用组合式函数来拆分逻辑。通过优化 ref 的使用,可以提升代码的可读性、性能和易维护性。


http://www.niftyadmin.cn/n/5667617.html

相关文章

C++解决n点最小曼哈顿距离

作者制作不易&#xff0c;关注、点赞、收藏一下吧&#xff01; 1.曼巴顿距离 ‌‌曼哈顿距离是由十九世纪的‌赫尔曼闵可夫斯基所创词汇‌&#xff0c;用于标明两个点在标准坐标系上的绝对轴距总和。 2.代码实现 2.1.导入头文件、命名空间 这个没有什么好说的&#xff0c;…

通信工程学习:什么是ODN光分配网络

ODN&#xff1a;光分配网络 ODN&#xff08;Optical Distribution Network&#xff0c;光分配网络&#xff09;是光接入网中的重要组成部分&#xff0c;它位于光线路终端&#xff08;OLT&#xff09;和光网络单元&#xff08;ONU&#xff09;/光网络终端&#xff08;ONT&#x…

【Python】耗时任务的超时管理

一、背景介绍 在日常编程中&#xff0c;我们经常会遇到一些耗时的任务&#xff0c;如文件处理、网络请求等。为了提高程序的执行效率&#xff0c;我们可以采用多进程的方式来实现任务的并行处理。然而&#xff0c;在某些情况下&#xff0c;任务执行时间过长可能会导致程序卡顿&…

ARM中要使用的汇编基础

汇编代码&#xff1a; preserve8 area reset, code, readonly code32 entry ; 异常向量表 b start ; 重置异常&#xff0c;跳转到 start 标签 ldr pc, do_undifined ; 未定义指令异常处理函数地址 ldr pc, do_swi ; 软件中断异常处理函数地…

初体验《SpringCloud 核心组件Eureka》

文章目录 1.案例准备1.1 案例说明1.2 案例数据库准备1.3 环境搭建1.3.1. 创建一个空的项目1.3.2. 创建Maven工程1.3.3. 配置父工程依赖&#xff0c;SpringCloud版本以及对应的SpringBoot版本1.3.4. 创建公共模块1.3.5. 创建用户模块工程1.3.5.1 引入依赖以及配置文件1.3.5.2 在…

Apollo(阿波罗)架构由浅入深剖析

1.最简架构 如果不考虑分布式微服务架构中的服务发现问题,Apollo 的最简架构如下图所示: 注意事项: ConfigService 是一个独立的微服务,服务于 Client 进行配置获取。 Client 和 ConfigService 保持长连接,通过一种拖拉结合 (push & pull) 的模式,实现配置实时更新…

Node-RED和物联网分析:实时数据处理和可视化平台

这篇论文的标题是《Node-RED and IoT Analytics: A Real-Time Data Processing and Visualization Platform》&#xff0c;发表在《Tech-Sphere Journal of Pure and Applied Sciences (TSJPAS)》2024年第一期上。论文主要探讨了Node-RED和物联网分析在物联网(IoT)实时数据处理…

开源 AI 智能名片链动 2+1 模式 S2B2C 商城小程序与社交电商的崛起

摘要&#xff1a;本文深入探讨了社交电商迅速发展壮大的原因&#xff0c;并分析了开源 AI 智能名片链动 21 模式 S2B2C 商城小程序在社交电商中的重要作用。通过对传统电商与社交电商的对比&#xff0c;以及对各发展因素的剖析&#xff0c;阐述了该小程序如何为社交电商提供新的…