【Vue】Vue2和Vue3响应式原理

前言

Vue 3 的核心部分可以分为三个主要模块:Compiler、Reactivity 和 Runtime。响应式的处理逻辑在 Reactivity 部分。

Compiler(编译器):Template => 渲染函数

将 Vue 的模板(Template)转换成 JavaScript 渲染函数。

在 Vue 3 中,编译器通过 @vue/compiler-sfc 提供的工具将 .vue 文件中的模板、脚本和样式拆分开,并编译为 JavaScript 代码。模板会被转换成渲染函数(render function),这个渲染函数会在 Runtime 中执行来生成虚拟 DOM。

Runtime(运行时)

负责执行渲染函数、更新虚拟 DOM、协调更新视图、处理生命周期钩子等。运行时并不涉及模板的编译,它负责 Vue 应用的整体生命周期和视图更新。

虚拟 DOM:渲染函数生成的虚拟 DOM 会通过 diff 算法与现有虚拟 DOM 对比,然后更新到真实 DOM。
组件生命周期:通过运行时机制控制组件的生命周期钩子(如 mounted, updated 等)。
事件处理:事件绑定、事件触发等操作也由 runtime 负责。

Reactivity(响应式系统)

实现 Vue 3 的响应式数据机制,即在数据发生变化时自动更新视图。Vue 3 引入了 Proxy 作为响应式的核心实现方式,替代了 Vue 2 中的 Object.defineProperty

关键点:

  • reactive:将对象变成响应式对象。
  • ref:将基本数据类型(如字符串、数字)包装成响应式对象。
  • effect:自动追踪依赖,当数据变化时自动更新视图。
  • computed:用于创建依赖其他响应式数据的计算属性。

Vue2对于响应式的处理

Vue2 使用 Object.defineProperty,缺点如下:

  • 针对对象的某个属性,劫持整个对象则需要遍历,性能不如Proxy,具体可看本人另一篇文章:为什么Proxy性能优于Object.defineProperty
  • 无法劫持对象 新增删除属性,需要单独重写原型方法

Vue3对于响应式的处理

Vue3:通过 Proxy和Reflect 对整个对象进行劫持,主要代码如下:

get(target,key,receiver){
  const result = Reflect.get(target,key,receiver)
  track(traget,key) // 追踪
  return isObject(result) ? reactive(result):result
},
set(target,key,value,receiver){
  const oldValue = target[key]
  const result = Reflect.set(target,key,value,receiver)
  if(oldValue !== value){
    trigger(target,key) // 更新
  }
  return result
}

track 响应式数据依赖追踪

在这里插入图片描述

trigger 触发更新

在这里插入图片描述

使用 Reflect 是为了解决 this 带来的问题,通过如下示例说明:

const obj = {
  a: 1,
  b: 2,
  get c() {
    console.log(this)
    return this.a + this.b
  },
}

const handler = new Proxy(obj, {
  get(target, key, receiver) {
    console.log("get", key)
    return target[key]
  },
})

handler.c 

在这里插入图片描述

这个示例中,明明访问器成员c也依赖了a和b,但是却无法正常劫持,这是因为 Proxy 只拦截 直接的属性访问,而不会拦截通过 访问器 方法(getter)调用的属性。

所以需要通过 Reflect.get 使用对象的原始方法 [[get]],确保访问器方法依赖属性的

const obj = {
  a: 1,
  b: 2,
  get c() {
    console.log(this)
    return this.a + this.b
  },
}

const handler = new Proxy(obj, {
  get(target, key, receiver) {
    console.log("get", key)
    return Reflect.get(target, key, receiver)
  },
})

handler.c

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

哪些人群适合考取 PostgreSQL 数据库 PGCM 证书?

#postgresql#,作为开源数据库领域的佼佼者,凭借其强大的功能和广泛的应用场景,吸引了大量数据库从业者的关注。它代表着持有者在PostgreSQL数据库管理、优化、安全和高可用性设计等方面的专家级技能。 PGCM证书适合那些具备扎实理论基础和一…

C++高级编程(9)

九、STL模板库 1.C函数模板 函数模板是一个独立于类型的函数,可产生函数特定类型的版本。通过对参数类型进行参数化,获取有相同形式的函数体。 它是一个通用函数,它可适应一定范围内的不同类型对象的操作。 函数模板将代表着不同类型的一组…

深圳世界之窗:文化与娱乐交织的旅游胜地

深圳世界之窗位于广东省深圳市南山区华侨城,是中国著名的缩微景区。它以弘扬世界文化为宗旨,将世界奇观、历史遗迹、民间歌舞表演、高科技游乐项目等融为一体,为游客打造出一个不出国门就能领略世界风情的旅游胜地。 从文化角度来看&#xff…

贪心day04(买卖股票的最佳时机)

1.买卖股票的最佳时机 题目链接:. - 力扣(LeetCode) 思路:我们其实只需遍历一篇就可以解决这个问题。首先我们定义一个min为无穷大值,再遍历只要有数字比min跟小我们就更改min的值就好,此时我们只需要找出…

ClickHouse创建账号和连接测试

在之前搭建ClickHouse的时候,把账户相关的去掉了,所以登录和连接的时候是不需要账号密码的,但是实际项目中,肯定是需要根据需要创建账号。 一,创建账号 1,进入到 /etc/clickhouse-server, 编辑…

网页版五子棋——匹配模块(客户端开发)

前一篇文章:网页版五子棋——用户模块(客户端开发)-CSDN博客 目录 前言 一、前后端交互接口设计 二、游戏大厅页面 1.页面代码编写 2.前后端交互代码编写 3.测试获取用户信息功能 结尾 前言 前面文章介绍完了五子棋项目用户模块的代码…

10. java基础知识(下)

文章目录 一、一带而过二、字符串类型String1. 简单了解2. 关于结束符\03. 自动类型转换与强制类型转换 三、API文档与import导包1. API文档2. import导包 四、java中的数组1. 创建2. 遍历3. 补充4. Arrays类① 简单介绍② 练习 五、方法的重载六、规范约束七、内容出处 一、一…

Sam Altman:年底将有重磅更新,但不是GPT-5!

大家好,我是木易,一个持续关注AI领域的互联网技术产品经理,国内Top2本科,美国Top10 CS研究生,MBA。我坚信AI是普通人变强的“外挂”,专注于分享AI全维度知识,包括但不限于AI科普,AI工…

《Python编程实训快速上手》第四天--字符串操作

一、处理字符串 1、单引号和双引号 Python中单双引号均可以表示字符串,区别在于: 1、双引号中可以使用到单引号 2、单引号字符串中如果要使用单引号,要使用到转义字符 \ \ \t \n \\ 原始字符串 在开始的引号前加r&#xf…

原生html+js输入框下拉多选带关闭模块完整案例

<!DOCTYPE html> <html> <head> <title>多选下拉框</title> <style> * { box-sizing: border-box; margin: 0; padding: 0; } .multi-select-container { position: relative; width: 300px; margin: 20px; font-family: Arial, sans-seri…

在 Ubuntu 操作系统上:改变 App 任务栏菜单的背景色

Ubuntu 官方默认的终端&#xff0c;与操作系统的主题 theme 无关。

【优选算法 — 滑动窗口】滑动窗口小专题(一)

长度最小的子数组 长度最小的子数组 题目解析&#xff1a; 对于示例一 对于剩下两种示例&#xff1a; 解法一&#xff1a;暴力枚举 把所有的子数组全部枚举出来&#xff0c;并且枚举出的每一个子数组求和判断&#xff0c;返回长度最小的子数组&#xff1b; 时间复杂度 &…

半波整流器原理

一、二极管不控整流 1.阻性负载 1.1.电路拓扑结构 电路只由交流源、二极管和电阻组成。最基本的带阻性负载的半波整流器如图所示。输入源为交流源&#xff0c;目标是使输出电压含有非零直流分量&#xff0c;负载为R。功率二极管只允许电流往一个方向流动。 1.2.工作模态分析…

yolov8涨点系列之引入CBAM注意力机制

文章目录 YOLOv8 中添加注意力机制 CBAM 具有多方面的好处特征增强与选择通道注意力方面空间注意力方面 提高模型性能计算效率优化&#xff1a; yolov8增加CBAM具体步骤CBAM代码(1)在__init.pyconv.py文件的__all__内添加‘CBAM’(2)conv.py文件复制粘贴CBAM代码(3)修改task.py…

Rust-AOP编程实战

文章本天成&#xff0c;妙手偶得之。粹然无疵瑕&#xff0c;岂复须人为&#xff1f;君看古彝器&#xff0c;巧拙两无施。汉最近先秦&#xff0c;固已殊淳漓。胡部何为者&#xff0c;豪竹杂哀丝。后夔不复作&#xff0c;千载谁与期&#xff1f; ——《文章》宋陆游 【哲理】文章…

基于 SSM(Spring + Spring MVC + MyBatis)框架构建电器网上订购系统

基于 SSM&#xff08;Spring Spring MVC MyBatis&#xff09;框架构建电器网上订购系统可以为用户提供一个方便快捷的购物平台。以下将详细介绍该系统的开发流程&#xff0c;包括需求分析、技术选型、数据库设计、项目结构搭建、主要功能实现以及前端页面设计。 需求分析 …

Docker部署Oracle 11g

1&#xff0c;拉取镜像&#xff1a; sudo docker pull registry.cn-hangzhou.aliyuncs.com/helowin/oracle_11gsudo docker images 2&#xff0c;启动一个临时容器&#xff0c;用于拷贝数据库文件&#xff0c;挂载到宿主主机&#xff0c;使数据持久化&#xff1a; sudo docke…

Linux操作系统:学习进程_了解并掌握进程的状态

对进程状态之间转换感到头疼&#xff0c;只听书本概念根本无法理解&#xff0c;死记硬背不是什么好的解决方法。只有进行底层操作去了解每一个进程状态&#xff0c;才能彻底弄清楚进程状态是如何转换的。 一、进程的各个状态 我们先从Linux内核数据结构来看&#xff1a; 每一个…

分布式环境下宕机的处理方案有哪些?

大家好&#xff0c;我是锋哥。今天分享关于【分布式环境下宕机的处理方案有哪些&#xff1f;】面试题。希望对大家有帮助&#xff1b; 分布式环境下宕机的处理方案有哪些&#xff1f; 1000道 互联网大厂Java工程师 精选面试题-Java资源分享网 在现代分布式系统中&#xff0c;…

接触角测量(Contact Angle Measurement)

接触角是指液滴在固体表面上的形态&#xff0c;特别是在气、液、固三相交界点处&#xff0c;气-液界面切线与固-液交界线之间的夹角。这个角度是衡量液体对固体表面润湿性的一个重要参数。接触角的大小可以反映液体与固体之间的相互作用强度&#xff0c;从而了解液体在固体表面…