vue2响应式数据原理
1. 核心原理
Vue 2 的响应式系统基于 Object.defineProperty,通过 依赖收集 和 派发更新 来实现数据的响应式
- 依赖收集:在读取数据时,记录哪些函数(或组件)依赖了该数据。
- 派发更新:在修改数据时,通知所有依赖该数据的函数(或组件)进行更新。
简单示例
// 模块内的变量
let activeFunc = nullfunction observe(obj) {for (const key in obj) {// 临时属性let internalValue = obj[key];// 事件集合let fns = new Set()Object.defineProperty(obj, key, {get: function() {// 依赖收集if(activeFunc){fns.add(activeFunc)}return internalValue;}set: function (val) {internalValue = val;// 如果新值是对象,递归调用 observeif (typeof val === 'object' && val !== null) {observe(val);}//派发更新fns.forEach(fn => fn());}});}
}// 定义vue内部一个模块变量
// 在调用“使用到响应式变量的dom”更新函数的时候,先将该函数赋值给该变量
// 这样vue在依赖收集的时候,就可以通过收集该变量,进而统一收集
function fn1 () {}
function fn2 () {}activeFunc = fn1 // fn表示更新vue模版的函数
fn1()activeFunc = fn2
fn2()
2. 优化后的完整示例
- 数组的响应式处理:
Object.defineProperty无法直接监听数组的变化(如push、pop等操作)。- Vue 2 通过重写数组的原型方法来实现数组的响应式。
- 依赖收集的封装:
- 将依赖收集和派发更新的逻辑封装成一个独立的
Dep类。
- 将依赖收集和派发更新的逻辑封装成一个独立的
activeFunc的管理:- 使用栈结构管理多个嵌套的依赖关系。
// 将依赖收集和派发更新的逻辑封装成一个独立的类
class Dep {constructor() {this.subscribers = new Set();}depend() {if (activeFunc) {this.subscribers.add(activeFunc);}}notify() {this.subscribers.forEach(fn => fn());}
}// 重写数组的原型方法
const arrayProto = Array.prototype;
const arrayMethods = Object.create(arrayProto);// Object.defineProperty 无法直接监听数组的变化(如 push、pop 等操作)
// Vue 2 通过重写数组的原型方法来实现数组的响应式
['push', 'pop', 'shift', 'unshift', 'splice', 'sort', 'reverse'].forEach(method => {const original = arrayProto[method];arrayMethods[method] = function (...args) {const result = original.apply(this, args);dep.notify(); // 派发更新return result;};
});// 当前正在执行的函数
let activeFunc = null;
// 用于管理嵌套的依赖关系
const activeFuncStack = [];// 将当前函数推入栈
function pushActiveFunc(fn) {activeFuncStack.push(fn);activeFunc = fn;
}// 将当前函数弹出栈
function popActiveFunc() {activeFuncStack.pop();activeFunc = activeFuncStack[activeFuncStack.length - 1];
}// 监听对象,将其属性转换为响应式
function observe(obj) {if (Array.isArray(obj)) {// 如果是数组,重写其原型方法obj.__proto__ = arrayMethods;// 递归监听数组元素obj.forEach(item => observe(item));} else if (typeof obj === 'object' && obj !== null) {for (const key in obj) {let internalValue = obj[key];const dep = new Dep();// 递归监听嵌套对象observe(internalValue);Object.defineProperty(obj, key, {get: function () {dep.depend(); // 依赖收集return internalValue;},set: function (val) {internalValue = val;observe(val); // 递归监听新值dep.notify(); // 派发更新}});}}
}// 副作用函数,用于追踪依赖
function watchEffect(fn) {pushActiveFunc(fn);fn(); // 执行函数,触发依赖收集popActiveFunc();
}// 测试
const data = { count: 0, list: [1, 2, 3] };
observe(data);watchEffect(() => {console.log('Count:', data.count);
});watchEffect(() => {console.log('List:', data.list);
});data.count++; // 输出: Count: 1
data.list.push(4); // 输出: List: [1, 2, 3, 4]
3. 关键点解析
- 依赖收集:
- 在
get方法中,通过dep.depend()收集当前正在执行的函数(activeFunc)。 activeFunc是通过watchEffect设置的,表示当前正在执行的副作用函数。
- 在
- 派发更新:
- 在
set方法中,通过dep.notify()通知所有依赖该属性的函数进行更新。
- 在
- 数组的响应式处理:
- 通过重写数组的原型方法,拦截数组的修改操作,并手动触发更新。
- 嵌套对象和数组的处理:
- 使用递归监听嵌套对象和数组,确保所有层级的属性都是响应式的。
4. 总结
Vue 2 的响应式系统通过 Object.defineProperty 实现数据的监听,虽然功能强大,但也存在一些局限性(如无法直接监听数组的变化)。通过重写数组方法和封装依赖收集逻辑,Vue 2 实现了完整的响应式系统。
相关文章:
vue2响应式数据原理
1. 核心原理 Vue 2 的响应式系统基于 Object.defineProperty,通过 依赖收集 和 派发更新 来实现数据的响应式 依赖收集:在读取数据时,记录哪些函数(或组件)依赖了该数据。派发更新:在修改数据时ÿ…...
OpenBMC:BmcWeb实例化App
BmcWeb是OpenBMC的一个核心模块,对外负责响应Redfish请求,并且由于OpenBMC的Web使用的Redfish api,所以BmcWeb也是Web的后台。 1.main函数 //src\webserver_main.cpp #include "webserver_run.hpp"int main(int /*argc*/, char** /*argv*/) noexcept(false) {re…...
二级公共基础之数据库设计基础(一) 数据库系统的基本概念
目录 前言 一、数据库、数据管理系统和数据库系统 1.数据 2.数据库 3.数据库管理系统 1.数据库管理系统的定义 2.数据库管理系统的功能 1.数据定义功能 2.数据操作功能 3.数据存取控制 4.数据完整性管理 5.数据备份和恢复 6.并发控制 4.数…...
自然语言处理NLP 04案例——苏宁易购优质评论与差评分析
上一篇文章,我们爬取了苏宁易购平台某产品的优质评价和差评,今天我们对优质评价与差评进行分析 selenium爬取苏宁易购平台某产品的评论-CSDN博客 目录 1. 数据加载 2. 中文分词 3. 停用词处理 4. 数据标注与合并 5. 数据集划分 6. 文本特征提取 …...
图片爬取案例
修改前的代码 但是总显示“失败” 原因是 修改之后的代码 import requests import os from urllib.parse import unquote# 原始URL url https://cn.bing.com/images/search?viewdetailV2&ccidTnImuvQ0&id5AE65CE4BE05EE7A79A73EEFA37578E87AE19421&thidOIP.TnI…...
leetcode_动态规划/递归 70. 爬楼梯
70. 爬楼梯 假设你正在爬楼梯。需要 n 阶你才能到达楼顶。 每次你可以爬 1 或 2 个台阶。你有多少种不同的方法可以爬到楼顶呢? 思路: 考虑: 假设现在已经爬到了某一阶台阶,那是如何到达这里的呢?可能是从前一阶台阶爬上来的&am…...
VoIP之音频3A技术
音频3A技术是改善语音通话质量的三种关键技术的简称,包括声学回声消除(Acoustic Echo Cancellation, AEC)、自动增益控制(Automatic Gain Control, AGC)、自噪声抑制(Automatic Noise Suppression, ANS&…...
SpringBoot 03 Web开发
SpringBoot 代替了SSM整合,避免了版本依赖冲突,不在需要手写配置文件 前言 springboot提供了starter场景启动器(web,Tomcat,jdbc),自带相关组件实现自动配置 场景启动器 一、自动配置 1.自动…...
官方文档学习TArray容器
一.TArray中的元素相等 1.重载一下 元素中的 运算符,有时需要重载排序。接下来,我们将id 作为判断结构体的标识。 定义结构体 USTRUCT() struct FXGEqualStructInfo {GENERATED_USTRUCT_BODY() public:FXGEqualStructInfo(){};FXGEqualStructInfo(in…...
Web刷题之PolarDN(中等)
1.到底给不给flag呢 代码审计 一道典型的php变量覆盖漏洞 相关知识 什么是变量覆盖漏洞 自定义的参数值替换原有变量值的情况称为变量覆盖漏洞 经常导致变量覆盖漏洞场景有:$$使用不当,extract()函数使用不当,parse_str()函数使用不当&…...
Https通信中证书验证流程
在 HTTPS 通信中,客户端验证服务器证书的有效性是确保通信安全的重要步骤。这一过程通常被称为 证书链验证 或 SSL/TLS 证书验证。以下是详细的流程和实现细节: 1. 证书验证的整体流程 客户端验证服务器证书的有效性主要包括以下几个步骤: …...
学习笔记-250222
论文: Learning Hierarchical Prompt with Structured Linguistic Knowledge for Vision-Language Models 主要研究llm在图像分类中的能力,当提示输入目标类别时,llm能够生成相关的描述以及相应的结构化关系。 1.首先利用llm从普通的描述中获…...
Unity游戏制作中的C#基础(1)界面操作基础
1.脚本有关注意事项 (1).进入项目之后,一般创建一个文件夹Scripts用来存放c#脚本; (2).在Scripts中创建脚本,双击脚本,进入VS编辑器,有如下结构: start&#…...
为什么要将PDF转换为CSV?CSV是Excel吗?
在企业和数据管理的日常工作中,PDF文件和CSV文件承担着各自的任务。PDF通常用于传输和展示静态的文档,而CSV因其简洁、易操作的特性,广泛应用于数据存储和交换。如果需要从PDF中提取、分析或处理数据,转换为CSV格式可能是一个高效…...
Android KMP初探
Android KMP初探 前言: 最近线上听了Kotlin官网举行的KMP会议,感觉听神奇的,于是就把官方demo下载下来尝试了一下,下载插件和所需要的依赖都用了很久,但是发现里面的代码很少,于是尝试自己手写了一下&…...
网络安全之Web后端PHP
目录 一、PHP基础语法 1.PHP基础 (1)php的优点 (2)PhpStorm的优点 2.PHP基本语法 3.PHP变量 4.PHP运算符 二、PHP流控与数组 1.php流程控制语句以及循环 (1)if 语句 (2)if…...
Redis——用户签到BitMap,UV统计
目录 BitMap 使用场景 1. 用户签到系统 2. 用户行为标记 3. 布隆过滤器(Bloom Filter) BitMap介绍 Redis中的使用 Redis功能示例 添加: 获取: 批量获取: java中实现 统计本月连续签到次数 UV统计 UV 统计…...
pycharm技巧--鼠标滚轮放大或缩小 Pycharm 字体大小
1、鼠标滚轮调整字体 设置 Ctrl 鼠标滚轮调整字体大小 备注: 第一个是活动窗口,即缩放当前窗口 第二个是所有编辑器窗口,即缩放所有窗口的字体 2、插件 汉化包: Chinese Simplified 包...
数字信任的底层逻辑:密码学核心技术与现实应用
安全和密码学 --The Missing Semester of Your CS Education 目录 熵与密码强度密码散列函数密钥体系 3.1 对称加密 3.2 非对称加密信任模型对比典型应用案例安全实践建议扩展练习杂项 密码学是构建数字信任的基石。 本文浅析密码学在现实工具中的应用,涵盖 1&…...
全面理解-深拷贝与浅拷贝
在 C 中,深拷贝(Deep Copy) 和 浅拷贝(Shallow Copy) 是两种完全不同的对象拷贝策略,主要区别在于对指针和动态分配资源的处理方式。正确理解二者的区别是避免内存泄漏、悬空指针和程序崩溃的关键。 一、核…...
Redis分布式锁故障处理:当Redis不可用时的应对策略
Redis分布式锁故障处理:当Redis不可用时的应对策略 在分布式系统中,Redis因其高性能和丰富的特性常被用于实现分布式锁。但当加锁过程中Redis服务不可用时,系统将面临严重挑战。本文将深入探讨这一问题,并提供多维度解决方案。 目…...
WordPress平台如何接入Deepseek,有效提升网站流量
深夜改代码到崩溃?《2024全球CMS生态报告》揭露:78%的WordPress站长因API对接复杂,错失AI内容红利。本文实测「零代码接入Deepseek」的保姆级方案,配合147SEO的智能发布系统,让你用3个步骤实现日均50篇EEAT合规内容自动…...
ROS ur10机械臂添加140夹爪全流程记录
ROS ur10机械臂添加140夹爪 系统版本:Ubuntu20.04 Ros版本:noetic Moveit版本:moveit-noetic 参考博客: ur3robotiq ft sensorrobotiq 2f 140配置rviz仿真环境_有末端力传感器的仿真环境-CSDN博客 UR5机械臂仿真实例…...
16、Python面试题解析:python中的浅拷贝和深拷贝
在 Python 中,浅拷贝(Shallow Copy) 和 深拷贝(Deep Copy) 是处理对象复制的两种重要机制,它们的区别主要体现在对嵌套对象的处理方式上。以下是详细解析: 1. 浅拷贝(Shallow Copy&a…...
第十九天 HarmonyOS的文件操作和本地存储
一、前言:为什么需要掌握文件操作与本地存储? 在移动应用开发中,文件操作和本地存储是每个开发者都必须掌握的核心技能。无论是保存用户配置、缓存网络数据,还是处理图片/视频等多媒体文件,都需要通过文件系统进行操作…...
VLM(视觉语言模型)与DeepSeek R1(奖励机制)如何结合
VLM(视觉语言模型)与DeepSeek R1(奖励机制)如何结合 flyfish VLM的传统训练依赖于监督学习(直接拟合问答对),而规则奖励函数通常用于强化学习(通过试错和奖励反馈优化策略…...
FFMPEG编码容错处理解决办法之途径----升级库文件
在qt开发环境下接收网络数据,调用ffmpeg解码播放视频,出现闪屏现象,具体现象可以使用操作系统自带的ffplay播放器播放原始视频流可复现;而使用操作系统自带的mpv播放器播放视频则不会出现闪屏;闪屏时会报Could not fin…...
uniapp h5端和app端 使用 turn.js
前提:添加页后,添加页与当前页会重叠在一起,不知道为什么,没有找到解决办法 1.h5端 <template><view class"container"><view id"flipbook"><view class"page page1">Page 1</view><view class"page pag…...
【idea问题排查技巧】
以下是针对 IDEA 中 日志打标(动态标记) 和 全链路追踪 功能的分步详解,结合具体场景和操作截图说明,帮助快速掌握实战技巧。 一、动态日志打标:不修改代码输出关键信息 1. 断点日志打印(非侵入式打标) 场景:在调试时,需要临时查看某个变量的值,但不想修改代码添加…...
【入门音视频】音视频基础知识
🌈前言🌈 这个系列在我学习过程中,对音视频知识归纳总结的笔记。因为音视频相关讲解非常稀少,所以我希望通过这个音视频系列,跟大家一起学习音视频,希望减少初学者在学习上的压力。同时希望也欢迎指出文章的…...
