vue3-ref 和 reactive

news2024/11/5 12:57:32

文章目录

    • vue3 中 ref 和 reactive
    • reactive 与 ref 不同之处
    • ref 处理复杂类型
    • ref在dom中的应用

vue3 中 ref 和 reactive

  1. ref原理
    • 基本原理
      • ref是Vue 3中用于创建响应式数据的一个函数。它的基本原理是通过Object.defineProperty()(在JavaScript的规范中用于定义对象属性的方法)或者Proxy(ES6新特性,用于创建对象的代理)来实现数据的响应式。当数据被修改时,它会触发更新相关的副作用函数(例如更新DOM等操作)。
      • 当使用ref创建一个响应式数据时,它实际上返回一个包含value属性的对象。例如,const count = ref(0),这里的count是一个对象,访问和修改数据需要通过count.value
    • 底层实现细节(简单示例)
      • 以下是一个简单的模拟ref实现的代码(简化版,实际Vue 3代码更复杂):
function ref(raw) {
    const r = {
        get value() {
            track(r, 'value');
            return raw;
        },
        set value(newVal) {
            if (newVal!== raw) {
                raw = newVal;
                trigger(r, 'value');
            }
        }
    };
    return r;
}
  • 这里的track函数用于收集依赖(记录哪些函数依赖了这个ref数据),trigger函数用于触发更新(当数据改变时,通知依赖的函数重新执行)。
  1. reactive原理
    • 基本原理
      • reactive是基于Proxy对象来实现响应式的。Proxy可以拦截对目标对象的各种操作,如读取属性、设置属性、删除属性等。当这些操作发生时,reactive可以触发相应的更新逻辑。
      • 例如,const state = reactive({ count: 0 }),对state.count的读取和修改都会被Proxy拦截,从而实现响应式。
    • 底层实现细节(简单示例)
      • 以下是一个简单的模拟reactive实现的代码片段(简化版):
function reactive(target) {
    return new Proxy(target, {
        get(target, key, receiver) {
            track(target, key);
            return Reflect.get(target, key, receiver);
        },
        set(target, key, value, receiver) {
            const result = Reflect.set(target, key, value, receiver);
            trigger(target, key);
            return result;
        }
    });
}
  • 同样,这里的tracktrigger函数用于收集依赖和触发更新,和ref的原理类似,但是reactive是针对整个对象的属性进行拦截。
  1. ref和reactive的应用场景

    • ref应用场景
      • 基本数据类型:当处理基本数据类型(如numberstringboolean)的响应式数据时,通常使用ref。例如,一个计数器const count = ref(0),这里count是一个ref对象,通过count.value来访问和修改计数值。
      • 在组合式API中传递单个数据:如果在组合式API函数之间传递一个简单的响应式数据,ref是一个很好的选择。比如一个函数返回一个ref数据,另一个函数可以接收并使用这个ref
    • reactive应用场景
      • 复杂对象类型:对于复杂的对象(如包含多个属性的对象或数组),使用reactive更为合适。例如,const state = reactive({ name: 'John', age: 30, hobbies: ['reading', 'coding']}),对state对象中的任何属性的访问和修改都会是响应式的。
      • 多个相关数据的管理:当需要管理一组相关的数据并且希望它们作为一个整体进行响应式处理时,reactive是首选。比如一个表单数据对象,包含表单的各个字段值,使用reactive可以方便地处理整个表单数据的更新和响应。
  2. 应用要求/依据

    • 响应式数据的修改原则
      • 对于ref,必须通过.value属性来修改数据才能触发响应式更新。例如,count.value++会触发更新,而直接count = new_value(这里countref)不会触发响应式更新。
      • 对于reactive,直接修改对象的属性就可以触发响应式更新。如state.name = 'Jane'(这里statereactive对象)会触发更新。
    • 模板中的使用差异
      • 在Vue 3的模板中,ref的数据可以直接在模板中使用,不需要额外的.value语法。例如,<div>{{ count }}</div>(这里countref)是可以正常工作的,Vue会自动处理value属性的访问。而对于reactive对象,直接通过属性访问即可,如<div>{{ state.name }}</div>(这里statereactive对象)。

reactive 与 ref 不同之处

  1. 数据类型处理方式不同
    • ref
      • 主要用于包装基本数据类型(如numberstringboolean),使其具有响应式特性。例如const count = ref(0)count是一个对象,它有一个value属性来存储实际的数据。在JavaScript代码中,必须通过count.value来访问和修改这个数据。
    • reactive
      • 用于处理复杂的对象类型(如对象字面量或数组),它会递归地将对象的所有属性转换为响应式。例如const state = reactive({ count: 0, message: 'Hello'})state本身就是响应式对象,对state.countstate.message等属性的访问和修改都是响应式的,不需要像ref那样通过特定的value属性来操作。
  2. 在模板中的使用差异
    • ref
      • 在Vue 3模板中使用ref时,不需要显式地访问value属性。Vue会自动进行解包,例如,如果count是一个ref,在模板中可以直接写成<div>{{ count }}</div>,它会正确地渲染count.value的值。
    • reactive
      • 对于reactive对象,在模板中直接通过属性访问来使用。如const state = reactive({ name: 'John'}),在模板中可以写成<div>{{ state.name }}</div>来显示name属性的值。
  3. 响应式转换的深度不同
    • ref
      • 只对它包装的那个数据本身进行响应式处理。如果ref包装了一个对象,它不会自动将对象内部的属性也转换为响应式,仍然需要通过value属性来访问对象,并且如果要使对象内部属性具有响应式,可能还需要进一步的处理(如使用reactive来处理这个对象)。
    • reactive
      • 是深度响应式的。当使用reactive处理一个对象时,它不仅会使对象本身的属性具有响应式,还会递归地将对象内部嵌套的对象和数组的属性也转换为响应式。例如,如果有一个对象const user = reactive({ name: 'Alice', address: { city: 'New York', street: '123 Main St' }}),那么user.address.city的访问和修改也是响应式的。
  4. 创建和使用的语法及场景差异
    • ref
      • 创建语法简单直接,适用于单个数据值的响应式处理,特别是基本数据类型。在组合式API中,当需要从一个函数返回一个响应式数据,并且这个数据是基本类型或者只需要简单的包装时,ref是很好的选择。
    • reactive
      • 创建时需要传入一个对象,更适合处理复杂的、有多个相关属性的数据结构,如应用程序中的状态对象。当有一组相关的数据需要作为一个整体进行响应式管理时,如表单数据、应用的全局状态等,reactive是更合适的方式。

ref 处理复杂类型

  1. ref处理复杂类型

    • 从功能上来说,ref可以用来包装复杂的数据类型,如对象或数组。例如,可以使用ref创建一个包含多个属性的对象,const myObject = ref({name: 'John', age: 30})
  2. 限制

    • 访问方式的限制
      • 在JavaScript代码中,必须通过value属性来访问和修改包装后的复杂对象。例如,要访问上述myObject中的name属性,需要使用myObject.value.name。这与reactive不同,reactive可以直接通过属性访问(如const myReactiveObject = reactive({name: 'Alice'}); myReactiveObject.name)。这种访问方式的差异可能会导致代码不够简洁直观,尤其是在处理多层嵌套的复杂对象时。
    • 响应式转换深度的限制
      • ref本身不会对包装的复杂对象内部进行深度的响应式转换。例如,如果有一个ref包装的对象const user = ref({name: 'Bob', address: {city: 'New York'}}),修改user.value.address.city不会自动触发响应式更新,除非address这个对象本身也经过了响应式处理(如使用reactive来处理address对象)。而reactive会自动对对象内部进行深度响应式处理。
    • 在模板中的限制(部分)
      • 在Vue 3的模板中,ref包装的复杂对象在某些情况下可能会引起混淆。虽然Vue会自动展开ref的值,但对于复杂对象的属性访问,有时候可能需要注意绑定的语法。例如,如果要在模板中使用myObject中的name属性,写成{{myObject.name}}是不行的,需要写成{{myObject.value.name}},不过Vue在模板渲染时会自动处理refvalue属性,使得在大多数简单情况下可以像使用普通数据一样使用ref包装的数据。但在复杂的模板绑定场景下,还是要注意正确的语法。

ref在dom中的应用

  1. 应用实例
    • 基本计数器示例
      • 在Vue 3组件中,可以使用ref来获取DOM元素,并对其进行操作。以下是一个简单的计数器组件,当点击按钮时,计数器的值会增加,并且会更新DOM中的显示内容。
<template>
  <div>
    <p ref="counterDisplay">{{ count }}</p>
    <button @click="increment">Increment</button>
  </div>
</template>

<script>
import { ref, defineComponent } from 'vue';
export default defineComponent({
  setup() {
    const count = ref(0);
    const counterDisplay = ref(null);
    const increment = () => {
      count.value++;
      if (counterDisplay.value) {
        // 直接操作DOM元素的文本内容
        counterDisplay.value.textContent = count.value;
      }
    };
    return {
      count,
      counterDisplay,
      increment
    };
  }
});
</script>
  • 表单验证示例
    • 假设有一个简单的表单,通过ref获取表单元素,用于检查表单是否有效。
<template>
  <form ref="myForm">
    <input type="text" v-model="username">
    <button @click="checkFormValidity">Check Validity</button>
  </form>
</template>

<script>
import { ref, defineComponent } from 'vue';
export default defineComponent({
  setup() {
    const username = ref('');
    const myForm = ref(null);
    const checkFormValidity = () => {
      if (myForm.value) {
        const isFormValid = myForm.value.checkValidity();
        console.log('Form is valid:', isFormValid);
      }
    };
    return {
      username,
      myForm,
      checkFormValidity
    };
  }
});
</script>
  1. 原理
    • 挂载阶段(mounted)
      • 当组件被挂载时,ref对应的DOM元素会被赋值。在Vue 3中,组件的setup函数执行过程中,ref初始化为null。然后,在组件的生命周期钩子(如onMounted)或者在模板渲染完成后(实际上是Vue内部的渲染机制),ref会被正确地赋值为对应的DOM元素。这是因为Vue在渲染过程中会遍历模板,识别出带有ref属性的DOM元素,并将其与对应的ref变量关联起来。
    • 更新阶段(update)
      • 当组件的状态发生变化(如count的值改变)并导致DOM需要更新时,Vue会重新渲染模板。在这个过程中,ref变量仍然保持对相应DOM元素的引用。如果在setup函数中有对ref变量所引用的DOM元素进行操作的代码(如修改textContent),这些操作会在DOM更新完成后或者在适当的时机(由Vue的响应式更新机制决定)执行。
    • 响应式原理关联
      • ref用于操作DOM元素也与Vue的响应式原理相关。当一个ref变量(如count)的值发生变化时,Vue会触发组件的重新渲染。在重新渲染过程中,其他与ref相关的操作(如操作counterDisplay所引用的DOM元素)可以利用这个更新时机来实现对DOM的动态操作。同时,ref本身的更新机制(通过value属性访问和修改)也保证了操作的一致性和可追踪性,使得在组件的整个生命周期内能够正确地对DOM进行操作。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2233578.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

使用Docker-Compose安装redis,rabbitmq,nacos,mysql,nginx,tomcat,portainer组件教程

因为开发经常会用到一些组件&#xff0c;又不想在本地启动&#xff0c;所以买了个服务器&#xff0c;然后将这些组件都安装到服务器上以便开发使用。下面就记录下使用docker-compose安装组件的教程以及一些需要注意的地方。 关于docker和docker-compose的安装在另一篇博客中有…

WPF+MVVM案例实战(二十二)- 制作一个侧边弹窗栏(CD类)

文章目录 1、案例效果1、侧边栏分类2、CD类侧边弹窗实现1、样式代码实现2、功能代码实现3 运行效果4、源代码获取1、案例效果 1、侧边栏分类 A类 :左侧弹出侧边栏B类 :右侧弹出侧边栏C类 :顶部弹出侧边栏D类 :底部弹出侧边栏2、CD类侧边弹窗实现 1、样式代码实现 在原有的…

解决 “Error: listen EACCES: permission denied 0.0.0.0:80“ 错误

前言 在开发过程中&#xff0c;我们经常会遇到各种各样的错误。其中一个常见的错误是 Error: listen EACCES: permission denied 0.0.0.0:80。这个错误通常发生在尝试启动一个开发服务器时&#xff0c;服务器试图绑定到80端口&#xff0c;但由于权限不足而失败。本文将详细介绍…

国产之光-海豚调度器的入门知识篇

目录 概念和定义 核心特性 核心组件 工作流程 环境准备 系统软件方面 硬件方面 部署方式 单机模式 伪集群模式 集群模式 基本配置 工作流定义 调度管理 定时调度 手动调度 监控告警 任务类型扩展 资源管理 权限控制 概念和定义 Apache DolphinScheduler是一…

使用Netty实现一个简单的聊天服务器

✅作者简介&#xff1a;热爱Java后端开发的一名学习者&#xff0c;大家可以跟我一起讨论各种问题喔。 &#x1f34e;个人主页&#xff1a;Hhzzy99 &#x1f34a;个人信条&#xff1a;坚持就是胜利&#xff01; &#x1f49e;当前专栏&#xff1a;Netty &#x1f96d;本文内容&a…

新世联科技:NG2-A-7在DAC空气捕集提取CO2的应用

一、DAC空气捕集提取CO2的介绍 直接空气碳捕获&#xff08;Direct Air Capture&#xff0c;简称DAC&#xff09;是一种直接从大气中提取二氧化碳的技术。 二、DAC空气捕集提取CO2的前景 从大气中提取的这种二氧化碳可以作为循环经济的一部分以各种不同方式使用。未来&#xf…

ISUP协议视频平台EasyCVR视频融合平台接入各类摄像机的方法

安防视频监控ISUP协议视频平台EasyCVR兼容性强、支持灵活拓展&#xff0c;平台可提供视频远程监控、录像、存储与回放、视频转码、视频快照、告警、云台控制、语音对讲、平台级联等视频能力。 想要将摄像机顺利接入EasyCVR平台&#xff0c;实现视频监控的集中管理和分发&#x…

(五)Spark大数据开发实战:灵活运用PySpark常用DataFrame API

目录 一、PySpark 二、数据介绍 三、PySpark大数据开发实战 1、数据文件上传HDFS 2、导入模块及数据 3、数据统计与分析 ①、计算演员参演电影数 ②、依次罗列电影番位前十的演员 ③、按照番位计算演员参演电影数 ④、求每位演员所有参演电影中的最早、最晚上映时间及…

达梦数据库宕机问题分析及处理

官方宕机原因排查 官方故障诊断排除 相关概念 达梦数据库宕机往往会产生core文件&#xff0c;解读core文件是分析宕机原因的主要手段&#xff0c;类似oracle的diag.trc或system dump转储文件&#xff0c;记录数据库线程状态、sql语句等。 首选的排查方向可以从内存溢出、磁盘…

spring ai 入门 之 结构化输出 - 把大模型llm返回的内容转换成java bean

目录 ​编辑 将AI非结构化文本转换为特定格式数据的应用场景说明 Spring AI 介绍 &#xff1a;为Java开发者打造的AI应用开发框架 Qwen 介绍 &#xff1a; 一个国内领先的开源大模型 Spring AI Alibaba框架介绍 &#xff1a; 一个国内最好的spring ai实现 使用spring ai …

HbuildderX运行到手机或模拟器的Android App基座识别不到设备 mac

寻找模拟器 背景&#xff1a; 运行的是h5&#xff0c;模拟器是网易MuMu。 首先检查一下是否配置dab环境&#xff0c;adb version 配置一下hbuilderX的adb&#xff1a; 将命令输出的路径配置到hbuilderx里面去&#xff0c;然后重启下HbuilderX。 开始安装基座…一直安装不…

使用Docker Compose构建多容器应用

&#x1f493; 博客主页&#xff1a;瑕疵的CSDN主页 &#x1f4dd; Gitee主页&#xff1a;瑕疵的gitee主页 ⏩ 文章专栏&#xff1a;《热点资讯》 使用Docker Compose构建多容器应用 引言 Docker Compose 简介 安装 Docker Compose 创建基本配置 运行多容器应用 查看服务状态 …

react-router与react-router-dom的区别

写法上的区别&#xff1a; 写法1: import {Swtich, Route, Router, HashHistory, Link} from react-router-dom;写法2: import {Switch, Route, Router} from react-router; import {HashHistory, Link} from react-router-dom;react-router实现了路由的核心功能 react-router-…

Python 字符串类型中 ``split(“\n“)`` 与 ``splitlines()`` 方法的一些区别

最近在以 self.__print("#" * 20 "\n") 调用自己写的 __print 接口时发现打印的时候 "\n" 没有打出来&#xff0c;进而发现了 split("\n") 与 splitlines() 方法的一些区别。 一个是参数上&#xff0c;split 需要传递一个字符串作为…

Java Iterator 实现杨辉三角

一、问题描述 杨辉三角定义如下&#xff1a; 1/ \1 1/ \ / \1 2 1/ \ / \ / \1 3 3 1/ \ / \ / \ / \1 4 6 4 1/ \ / \ / \ / \ / \ 1 5 10 10 5 1 把每一行看做一个list&#xff0c;试写一个 Iterator&#xff0c;不断输出下一行的 list&#xf…

Spark 的介绍与搭建:从理论到实践

目录 一、分布式的思想 &#xff08;一&#xff09;存储 &#xff08;二&#xff09;计算 二、Spark 简介 &#xff08;一&#xff09;发展历程 &#xff08;二&#xff09;Spark 能做什么&#xff1f; &#xff08;三&#xff09;spark 的组成部分 &#xff08;四&…

Linux操作系统 ------(3.文本编译器Vim)

目录 1.前言 2.本章学习目标 3.vim的三种工作模式 3.1一般模式‌ 3.2编辑模式‌ 3.3命令行模式‌ 4.运行vim 5.vim 不同工作模式下的常见命令 6.一般模式下的功能键 6.1移动光标类 6.2删除、复制和粘贴类 6.3查找替换类 7.从一般模式进入编辑模式 8.命令行模式下的…

RocketMQ的消息类型

RocketMQ的消息类型 文章目录 RocketMQ的消息类型一、顺序消息二、广播消息应用场景&#xff1a;示例代码&#xff1a;实现思路&#xff1a;注意点&#xff1a; 三、延时消息应用场景&#xff1a;核心方法&#xff1a; 四、批量消息应用场景&#xff1a;示例代码&#xff1a;注…

Selective Generation for Language Models 语言模型的选择性生成

生成式语言模型&#xff08;Generative Language Models, GLMs&#xff09;在文本生成任务中取得了显著进展。然而&#xff0c;生成内容的“幻觉”现象&#xff0c;即生成内容与事实或真实语义不符的问题&#xff0c;仍是GLMs在实际应用中的一个重大挑战。为了解决这一问题&…

git clone,用https还是ssh

前言 在使用Git去克隆项目时&#xff0c;会遇到https和ssh等形式&#xff0c;这两种又有何种区别呢&#xff0c;本文将重点讨论在具体使用中的问题。 注:第一次使用Git 时&#xff0c;需要先设置全局用户名和邮箱&#xff0c;否则后续使用命令时会报错&#xff0c;也是提醒先添…