Vue3之响应式系统详解

news2024/12/16 19:33:22

Vue3中的响应式系统是其核心功能之一,它使得数据变化能够自动触发视图更新,从而简化了开发过程,提高了开发效率。本文将详细阐述Vue3中的响应式系统,包括其核心概念、工作原理、实现方式、应用场景以及优势。同时,本文将提供对应的代码示例,以帮助读者更好地理解Vue3的响应式系统。

一、核心概念

Vue3的响应式系统基于Proxy对象和Reflect API实现,通过拦截对象属性的读写操作,来追踪数据的变化,并自动触发依赖的更新。以下是Vue3响应式系统的核心概念:

1. Proxy对象

Proxy是ES6引入的一种功能,它允许开发者创建一个对象的代理,通过拦截并重新定义基本操作(如属性访问、赋值、枚举等),来实现对对象的自定义行为。Vue3利用Proxy对象来拦截对象属性的读写操作,从而实现了响应式的数据绑定。

2. Reflect API

Reflect API是ES6引入的一种用于操作对象的API,它提供了一系列与Object对象相同的方法,但行为更加统一和一致。Vue3在拦截对象属性的读写操作时,使用了Reflect API来实现对原始操作的调用,从而保证了代码的简洁和一致性。

3. 响应式数据

响应式数据是指能够被Vue3的响应式系统追踪和响应的数据。在Vue3中,响应式数据通常是通过reactive函数或ref函数创建的。响应式数据发生变化时,依赖于这些数据的视图和计算属性会自动更新。

4. 依赖收集与触发更新

Vue3的响应式系统通过依赖收集来追踪哪些视图或计算属性依赖于某个响应式数据。当响应式数据发生变化时,响应式系统会触发依赖于这些数据的视图和计算属性的更新。

二、工作原理

Vue3的响应式系统通过拦截对象属性的读写操作,来追踪数据的变化,并自动触发依赖的更新。以下是Vue3响应式系统的工作原理:

1. 创建响应式对象

在Vue3中,响应式对象通常是通过reactive函数或ref函数创建的。这些函数会返回一个代理对象,该代理对象会拦截对象属性的读写操作。

import { reactive, ref } from 'vue';

const state = reactive({
  count: 0
});

const count = ref(0);

2. 拦截对象属性的读写操作

当响应式对象的属性被访问或修改时,代理对象会拦截这些操作,并调用相应的处理函数。这些处理函数会利用Reflect API来执行原始操作,并同时触发依赖的收集或更新。

const handler = {
  get(target, key, receiver) {
    // 依赖收集
    // ...
    return Reflect.get(target, key, receiver);
  },
  set(target, key, value, receiver) {
    // 触发更新
    // ...
    return Reflect.set(target, key, value, receiver);
  }
};

const proxyState = new Proxy(state, handler);

3. 依赖收集

当响应式对象的属性被访问时,响应式系统会进行依赖收集。它会在当前的作用域中查找是否有依赖该属性的副作用函数(如计算属性、渲染函数等),并将这些副作用函数存储在依赖集合中。

let activeEffect = null;

function effect(fn) {
  activeEffect = fn;
  fn();
  activeEffect = null;
}

effect(() => {
  console.log(proxyState.count); // 依赖收集
});

4. 触发更新

当响应式对象的属性被修改时,响应式系统会触发依赖于该属性的副作用函数的执行。这些副作用函数会重新计算或重新渲染依赖于该属性的视图和计算属性。

proxyState.count++; // 触发更新

三、实现方式

Vue3的响应式系统主要通过Proxy对象和Reflect API实现,同时结合了一些辅助函数和数据结构来追踪依赖和触发更新。以下是Vue3响应式系统的实现方式:

1. Proxy对象与Reflect API

Vue3使用Proxy对象来拦截对象属性的读写操作,并通过Reflect API来执行原始操作。这样可以在不修改原始对象属性的情况下,实现对数据变化的追踪和响应。

const handler = {
  get(target, key, receiver) {
    // 依赖收集
    track(target, key);
    return Reflect.get(target, key, receiver);
  },
  set(target, key, value, receiver) {
    // 触发更新
    trigger(target, key, value);
    return Reflect.set(target, key, value, receiver);
  }
};

const proxyState = new Proxy(state, handler);

2. 依赖收集与触发更新的辅助函数

Vue3提供了一些辅助函数来帮助实现依赖收集和触发更新。例如,track函数用于依赖收集,它会将当前的作用域和属性名存储在一个全局的依赖映射表中;trigger函数用于触发更新,它会从依赖映射表中获取依赖于该属性的副作用函数,并执行它们。

const targetMap = new WeakMap();

function track(target, key) {
  if (activeEffect) {
    let depsMap = targetMap.get(target);
    if (!depsMap) {
      targetMap.set(target, (depsMap = new Map()));
    }
    let deps = depsMap.get(key);
    if (!deps) {
      depsMap.set(key, (deps = new Set()));
    }
    deps.add(activeEffect);
  }
}

function trigger(target, key, value) {
  const depsMap = targetMap.get(target);
  if (depsMap) {
    const deps = depsMap.get(key);
    if (deps) {
      deps.forEach(effect => {
        effect();
      });
    }
  }
}

3. 响应式数据的数据结构

Vue3中的响应式数据通常是通过reactive函数或ref函数创建的。这些函数会返回一个代理对象,该代理对象会拦截对象属性的读写操作。同时,Vue3还会使用一些数据结构来追踪依赖和存储响应式数据的信息。

  • reactive函数返回的代理对象会拦截对象属性的读写操作,并通过targetMap来追踪依赖。
  • ref函数返回的响应式引用会包含一个value属性来存储实际的值,并通过内部的_object属性来追踪依赖。

四、应用场景

Vue3的响应式系统在实际项目中有着广泛的应用场景,它使得数据变化能够自动触发视图更新,从而简化了开发过程,提高了开发效率。以下是一些常见的应用场景:

1. 数据绑定与自动更新

Vue3的响应式系统使得数据变化能够自动触发视图更新,从而实现了数据绑定与自动更新。开发者只需将数据定义为响应式的,并将其绑定到视图上即可,无需手动操作DOM。

<template>
  <div>{{ state.count }}</div>
  <button @click="increment">Increment</button>
</template>

<script>
import { reactive } from 'vue';

export default {
  setup() {
    const state = reactive({
      count: 0
    });

    function increment() {
      state.count++;
    }

    return {
      state,
      increment
    };
  }
};
</script>

2. 计算属性

Vue3中的计算属性是基于响应式数据动态计算的值。当计算属性所依赖的响应式数据发生变化时,计算属性会自动重新计算。

<template>
  <div>{{ doubleCount }}</div>
  <button @click="increment">Increment</button>
</template>

<script>
import { reactive, computed } from 'vue';

export default {
  setup() {
    const state = reactive({
      count: 0
    });

    const doubleCount = computed(() => state.count * 2);

    function increment() {
      state.count++;
    }

    return {
      doubleCount,
      increment
    };
  }
};
</script>

3. 侦听器

Vue3中的侦听器可以用于监听响应式数据的变化,并在变化时执行相应的逻辑处理。例如,可以使用侦听器来执行异步操作或手动控制数据更新。

<template>
  <div>
    <p>{{ count }}</p>
    <button @click="increment">Increment</button>
  </div>
</template>

<script>
import { ref, watch } from 'vue';

export default {
  setup() {
    // 定义一个响应式数据
    const count = ref(0);

    // 定义一个方法用于增加count的值
    function increment() {
      count.value++;
    }

    // 使用侦听器监听count的变化
    watch(count, (newCount, oldCount) => {
      // 当count的值发生变化时,执行此回调
      console.log(`Count changed from ${oldCount} to ${newCount}`);

      // 这里可以执行任何你想要的逻辑,比如异步操作
      // 例如,假设我们有一个异步函数fetchData,它依赖于count的值
      // fetchData(newCount).then(data => {
      //   // 处理异步返回的数据
      // });
    });

    // 返回需要在模板中使用的数据和方法
    return {
      count,
      increment
    };
  }
};
</script>

在这个示例中,我们定义了一个响应式数据count,以及一个用于增加count值的方法increment。然后,我们使用watch函数来监听count的变化。当count的值发生变化时,watch函数的回调会被调用,并传入新的值newCount和旧的值oldCount。

侦听器在Vue3中非常有用,特别是当你需要在数据变化时执行一些副作用逻辑(如异步请求、手动DOM操作等)时。通过使用侦听器,你可以确保这些逻辑只在数据实际发生变化时才执行,从而提高了代码的效率和可维护性。

此外,watch函数还提供了更多的选项,如immediate和deep,允许你更精细地控制侦听器的行为。例如,immediate: true可以在侦听器创建时立即执行回调,而deep: true则可以用于监听对象内部属性的变化。这些选项使得Vue3的侦听器功能更加灵活和强大。

五、总结

Vue3的响应式系统是其核心,利用Proxy和Reflect API追踪数据变化,自动触发视图更新,简化开发。它包含Proxy对象拦截操作、Reflect API执行原始操作、响应式数据创建、依赖收集与触发更新等核心概念。工作原理是通过拦截属性读写,实现依赖收集,并在数据变化时触发更新。实现上,结合辅助函数和数据结构追踪依赖。应用场景广泛,如数据绑定、计算属性、侦听器等,提高了开发效率。侦听器可监听数据变化,执行逻辑处理,支持异步操作和精细控制,使代码更高效、可维护。

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

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

相关文章

多进程并发跑程序:pytest-xdist记录

多进程并发跑程序&#xff1a;pytest-xdist记录 pytest -s E:\testXdist\test_dandu.py pytest -s testXdist\test_dandu.py pytest -s &#xff1a;是按用例顺序依次跑用例 pytest -vs -n auto E:\testXdist\test_dandu.py pytest -vs -n auto&#xff0c;auto表示以全部进程…

网络层IP协议(TCP)

IP协议&#xff1a; 在了解IP协议之前&#xff0c;我们市面上看到的"路由器"其实就是工作在网络层。如下图&#xff1a; 那么网络层中的IP协议究竟是如何发送数据包的呢&#xff1f; IP报头&#xff1a; IP协议的报头是比较复杂的&#xff0c;作为程序猿只需要我们重…

前端传入Grule,后端保存到 .grl 文件中

前端传入Grule&#xff0c;后端保存到 .grl 文件中 通过简单的输入框&#xff0c;将Grule的部分拆解成 规则名称 规则描述 规则优先级 规则条件 规则逻辑Grule关键字 when Then 模拟了 if 判断的条件和逻辑部分 类似于 shell 和 ruby 之类的脚本语言&#xff0c;有 then 关键字…

vlan和vlanif

文章目录 1、为什么会有vlan的存在2、vlan(虚拟局域网)1、vlan原理1. 为什么这样划分了2、如何实现不同交换机相同的vlan实现互访呢3、最优化的解决方法&#xff0c;vlan不同交换机4、vlan标签和vlan数据帧 5、vlan实现2、基于vlan的划分方式1、基于接口的vlan划分方式2、基于m…

遗传算法与深度学习实战(27)——进化卷积神经网络

遗传算法与深度学习实战&#xff08;27&#xff09;——进化卷积神经网络 0. 前言1. 自定义交叉算子2. 自定义突变操作符3. 进化卷积神经网络小结系列链接 0. 前言 DEAP toolbox 中提供的标准遗传操作符对于自定义的网络架构基因序列来说是不够的。这是因为任何标准的交叉算子…

react-dnd 拖拽事件与输入框的文本选中冲突

问题描述 当我们使用拖拽库的时候&#xff0c;往往会遇到拖拽的一个元素他的子孙元素有输入框类型的dom节点&#xff0c;当拖拽的事件绑定在该元素身上时候&#xff0c;发现子孙的输入框不能进行文本选中了&#xff0c;会按住鼠标去选中文本的时候会触发拖拽 实际的效果&…

经典NLP案例 | 推文评论情绪分析:从数据预处理到模型构建的全面指南

NLP经典案例&#xff1a;推文评论情绪提取 项目背景 “My ridiculous dog is amazing.” [sentiment: positive] 由于所有推文每秒都在传播&#xff0c;很难判断特定推文背后的情绪是否会影响一家公司或一个人的品牌&#xff0c;因为它的病毒式传播&#xff08;积极&#xff0…

杨振宁大学物理视频中黄色的字,c#写程序去掉(原版改进,三)

上一节&#xff0c;我们分清了主次矛盾&#xff0c;并搞定了主要矛盾&#xff08;去掉黄色的字&#xff09;&#xff0c;这一节解决次要矛盾&#xff08;矩形色带&#xff09;。 我们的想法如图&#xff1a; 1&#xff0c;我们找到稳定黄色的最左边&#xff0c;最右边两点&…

第24周:文献阅读

目录 摘要 Abstract 一、现有问题 二、提出方法 三、创新点 模型结构创新 强化学习与GAN结合 属性特征与通顺性优化 四、方法论 生成对抗网络&#xff08;GAN&#xff09; 强化学习&#xff08;RL&#xff09; 模型组件 五、实验研究 数据集 数据预处理 评价指…

SQL server学习05-查询数据表中的数据(上)

目录 一&#xff0c;基本格式 1&#xff0c;简单的SQL查询语句 2&#xff0c;关键字TOP 3&#xff0c;关键字DISTINCT 二&#xff0c;模糊查询 1&#xff0c;通配符 三&#xff0c;对结果集排序 1&#xff0c;不含关键字DISTINCT 2&#xff0c;含关键字DISTINCT 3&…

【Azure 架构师学习笔记】- Azure Function (1) --环境搭建和背景介绍

本文属于【Azure 架构师学习笔记】系列。 本文属于【Azure Function 】系列。 前言 随着无服务计算的兴起和大数据环境中的数据集成需求&#xff0c; 需要使用某些轻量级的服务&#xff0c;来实现一些简单操作。因此Azure Function就成了微软云上的一个必不可少的组成部分。 …

KeepAlive与RouterView缓存

参考 vue动态组件&#xff1c;Component&#xff1e;与&#xff1c;KeepAlive&#xff1e; KeepAlive官网介绍 缓存之keep-alive的理解和应用 Vue3Vite KeepAlive页面缓存问题 vue多级菜单(路由)导致缓存(keep-alive)失效 vue3 router-view keeperalive对于同一路径但路径…

Linux:进程(环境变量、程序地址空间)

目录 冯诺依曼体系结构 操作系统 设计操作系统的目的 操作系统的管理 进程 PCB fork 进程状态 进程状态查看 僵尸进程 孤儿进程 进程优先级 查看、修改进程优先级命令 竞争、独立、并行、并发 进程切换 活动队列和运行队列 活动队列 过期队列 active指针…

希迪智驾持续亏损8.2亿:毛利率下滑,冲刺“自动驾驶矿卡第一股”

《港湾商业观察》黄懿 近日&#xff0c;希迪智驾&#xff08;湖南&#xff09;股份有限公司&#xff08;下称“希迪智驾”&#xff09;向港交所主板递交上市申请&#xff0c;联席保荐人为中金公司、中信建投国际、中国平安资本&#xff08;香港&#xff09;。 资料显示&#…

Rust之抽空学习系列(三)—— 编程通用概念(中)

Rust之抽空学习系列&#xff08;三&#xff09;—— 编程通用概念&#xff08;中&#xff09; 1、变量&可变性 在Rust中&#xff0c;变量默认是不可变的 fn main() {let x 5;println!("x is {}", x); }使用let来声明一个变量&#xff0c;此时变量默认是不可变…

OpenCV中的识别图片颜色并绘制轮廓

一、实验原理 使用OpenCV库在图像中识别和绘制特定颜色&#xff08;黄色&#xff09;的轮廓 二、实验代码 import cv2 import numpy as np# 读取图片并调整大小 img cv2.imread(./color_1.png) img cv2.resize(img,(600,600))# 将图片从BGR颜色空间转换到HSV颜色空间 img_h…

【Qt】qt基础

目录 一、使用Qt Creator创建qt项目 二、项目文件解析 三、Qt中创建图形化界面的程序的两种方法 四、对象树 五、Qt中处理打印乱码问题的利器&#xff1a;qDebug() 一、使用Qt Creator创建qt项目 1.选择项目模板 选中第一类模板Application(Qt应用程序&#xff0c;包含普…

Transformer入门(6)Transformer编码器的前馈网络、加法和归一化模块

文章目录 7.前馈网络8.加法和归一化组件9.组合所有编码器组件构成完整编码器 7.前馈网络 编码器块中的前馈网络子层如下图所示&#xff1a; 图1.32 – 编码器块 前馈网络由两个带有ReLU激活函数的全连接层组成。全连接层&#xff08;Fully Connected Layer&#xff09;有时也…

AI智算-k8s部署大语言模型管理工具Ollama

文章目录 简介k8s部署OllamaOpen WebUI访问Open-WebUI 简介 Github&#xff1a;https://github.com/ollama/ollama 官网&#xff1a;https://ollama.com/ API&#xff1a;https://github.com/ollama/ollama/blob/main/docs/api.md Ollama 是一个基于 Go 语言开发的可以本地运…

HTML/CSS总结

HTML 1.1 标题标签h 为了使网页更具有语义化&#xff0c;我们经常会在页面中用到标题标签&#xff0c;HTML提供了6个等级的标题&#xff0c;即 标题标签语义&#xff1a; 作为标题使用&#xff0c;并且依据重要性递减 其基本语法格式如下&#xff1a; <h1> 标题文本…