深入浅出理解vue2/vue3响应式原理

一、简介

当谈论Vue 2和Vue 3的响应式原理时,我们主要关注的是其数据双向绑定的机制。数据双向绑定是指当数据发生变化时,视图会自动更新;反之,当视图发生变化时,数据也会相应地更新。这种特性让我们在前端开发中更加高效地处理数据和用户界面。

二、vue2响应式原理

1、Vue 2的响应式原理示例

Vue 2的响应式原理: Vue 2使用了Object.defineProperty来实现响应式。在Vue 2中,当我们创建Vue实例时,它会遍历data选项中的所有属性,并使用Object.defineProperty将它们转换为getter和setter。这样一来,每当我们读取或修改data中的属性时,Vue都能捕获到这个操作,并触发视图的更新。

举个例子,假设我们有如下的Vue 2示例:

<div id="app"><p>{{ message }}</p><button @click="changeMessage">Change Message</button>
</div><script>const vm = new Vue({el: '#app',data: {message: 'Hello, Vue 2!'},methods: {changeMessage() {this.message = 'Hello, World!';}}});
</script>

在这个例子中,我们在data选项中定义了一个message属性,然后在视图中使用了{{ message }}来显示这个属性的值。当点击按钮时,changeMessage方法会被调用,将message属性的值改为’Hello, World!'。由于message属性已被Vue劫持,它会触发对应的setter,从而通知视图进行更新。

2、vue2手写简易版的响应式原理

代码演示部分:

// 简化版的观察者类
class SimpleWatcher {constructor(vm, key, updateFn) {this.vm = vm;this.key = key;this.updateFn = updateFn;// 在这里模拟一下Vue的依赖收集Dep.target = this;this.vm[this.key];Dep.target = null;}// 依赖更新时触发的方法update() {this.updateFn.call(this.vm, this.vm[this.key]);}
}// 简化版的依赖管理类
class SimpleDep {constructor() {this.subscribers = [];}// 添加观察者addSubscriber(subscriber) {this.subscribers.push(subscriber);}// 通知观察者进行更新notify() {this.subscribers.forEach((subscriber) => subscriber.update());}
}// 简化版的Vue响应式类
class SimpleVue {constructor(data) {this._data = data;this._proxyData(data);}// 将data对象转换为getter和setter_proxyData(data) {for (let key in data) {if (Object.prototype.hasOwnProperty.call(data, key)) {const dep = new SimpleDep();Object.defineProperty(data, key, {get: () => {if (Dep.target) {dep.addSubscriber(Dep.target);}return data['_'+key];},set: (newValue) => {data['_'+key] = newValue;dep.notify();},});}}}
}// Dep类,用于简化依赖收集
class Dep {static target = null;
}

运用:

// 使用示例
const data = {message: 'Hello, Vue!'
};const vm = new SimpleVue(data);// 添加观察者
new SimpleWatcher(vm, 'message', (value) => {console.log('数据更新了:', value);
});// 修改数据,触发更新
data.message = 'Hello, World!';

三、vue3响应式原理

手写简易版的响应式原理: 现在,让我们一步步手写一个简易版的Vue响应式系统。我们将使用JavaScript的Proxy对象来实现。Proxy是ES6引入的新特性,它可以拦截对对象的操作,包括读取、设置等,非常适合用来实现响应式。

<div id="app"><p>{{ message }}</p><button onclick="changeMessage()">Change Message</button>
</div><script>function reactive(data) {return new Proxy(data, {get(target, key) {console.log('读取数据', key);return target[key];},set(target, key, value) {console.log('更新数据', key, value);target[key] = value;updateView(); // 数据更新后,手动更新视图return true;}});}function updateView() {const messageElement = document.querySelector('p');messageElement.textContent = app.message;}const data = {message: 'Hello, Simple Vue!'};const app = reactive(data);function changeMessage() {app.message = 'Hello, World!';}// 页面加载完成后,手动更新视图updateView();
</script>

这里我们使用了reactive函数来将data对象转换为响应式对象。然后我们用Proxy对象对这个响应式对象进行拦截,实现了对属性的读取和设置操作的监听。当数据发生变化时,我们手动调用updateView函数来更新视图。

四、总结

Vue 2和vue3 简易版响应式原理对比: 虽然我们手写的简易版响应式原理不如Vue 2的实现复杂和完善,但基本思想是一致的。Vue 2使用Object.defineProperty拦截属性的读取和设置操作,而我们使用Proxy来达到同样的效果。Vue 2和我们的简易版响应式原理都利用了JavaScript的特性,实现了数据双向绑定的效果。Vue 2的实现更加完善,支持更多的特性和优化。

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

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

相关文章

Swift 如何确定 scrollView 已经滑动结束

在 iOS 的 UIScrollView 中&#xff0c;你可以通过实现 UIScrollViewDelegate 的方法来检测滑动结束事件。具体来说&#xff0c;你可以实现以下方法&#xff1a; func scrollViewDidEndDragging(_ scrollView: UIScrollView, willDecelerate decelerate: Bool) {if !decelerat…

Python 实现接口类的两种方式+邮件提醒+动态导入模块+反射(参考Django中间件源码)

实现抽象类的两种方式 方式一 from abc import ABCMeta from abc import abstractmethodclass BaseMessage(metaclassABCMeta):abstractmethoddef send(self,subject,body,to,name):pass 方式二 class BaseMessage(object):def send(self, subject, body, to, name):raise …

vue3使用响应式数据 + v-model导致响应式失效el-form表单无法输入的问题

文章目录 vue3使用响应式数据 v-model导致响应式失效el-form表单无法输入的问题 vue3使用响应式数据 v-model导致响应式失效el-form表单无法输入的问题 参考文章 重构vue2项目时发现的问题&#xff0c;原始项目使用的是Element-ui。 其实vue3可以使用适配的Element-plus 问…

EasyExcel读取多sheet excel异常

EasyExcel读取多sheet excel异常 场景 使用Spring Boot集成EasyExcel进行导入excel多sheet&#xff0c;执行异常 代码 file是MultipartFile&#xff0c;入参&#xff0c;主要处理流程如下 InputStream is file.getInputStream(); EasyExcel.read(is, SysDict.class, new …

Set集合类详解(附加思维导图)

目录 一、Set集合思维导图 二、set集合类常用方法 2.1、HashSet集合常用方法 2.2、TreeSet集合的使用 三、HashSet、LinkedHashSet、TreeSet的使用场景 四、list和set集合的区别 一、Set集合思维导图 二、set集合类常用方法 2.1、HashSet集合常用方法 ①&#xff1a;add…

redis面试1

Redis基础面试题 1、为什么要使用Redis做缓存 缓存的好处 使用缓存的目的就是提升读写性能。而实际业务场景下&#xff0c;更多的是为了提升读性能&#xff0c;带来更好的性 能&#xff0c;带来更高的并发量。Redis 的读写性能比 Mysql 好的多&#xff0c;我们就可以把 Mysq…

scrapy分布式+指纹去重原理

1&#xff0c;指纹去重原理存在于 scrapy.util.requests 里面 需要安装的包 pip install scrapy-redis-cluster # 安装模块 pip install scrapy-redis-cluster0.4 # 安装模块时指定版本 pip install --upgrade scrapy-redis-cluster # 升级模块版本 2&#xff0c;setting配置 …

【小白必看】利用Python生成个性化名单Word文档

文章目录 前言所需文件及文件格式说明excel数据如下word 模板如下文件目录格式及生成后的文件 导入所需的模块&#xff1a;打开 Excel 文件&#xff1a;选择工作表&#xff1a;获取数据列表&#xff1a;遍历数据并生成 Word 文档&#xff1a;完整代码结束语 前言 对于需要批量…

【MyBatis 学习一】认识MyBatis 第一个MyBatis查询

目录 一、认识MyBatis 1、MyBatis是什么&#xff1f; 2、为什么要学习MyBatis? 二、配置MyBatis环境 1、建库与建表 2、创建新项目 3、xml文件配置 &#xff08;1&#xff09;配置数据库连接 &#xff08;2&#xff09;配置 MyBatis 中的 XML 路径 三、测试&#x…

mysql(三)用户权限管理

目录 前言 一、概述 二、用户权限类型 三、用户赋权 四、权限删除 五、删除用户 前言 为什么要设置用户权限&#xff1f; MySQL设置用户管理权限的主要目的是为了确保数据库的安全性和数据的机密性。以下是一些原因&#xff1a; 1. 安全性&#xff1a;MySQL是一个开源的关系型…

Java集合之List

ArrayLsit集合 ArrayList集合的特点 ArrayList集合的一些方法 ①.add(Object element) 向列表的尾部添加指定的元素。 ②.size() 返回列表中的元素个数。 ③.get(int index) 返回列表中指定位置的元素&#xff0c;index从0开始。 public class Test {public static void m…

dependency walker工具简介及使用

dependency walker工具 简介使用 简介 官方概述&#xff1a; Dependency Walker is a free utility that scans any 32-bit or 64-bit Windows module (exe, dll, ocx, sys, etc.) and builds a hierarchical tree diagram of all dependent modules. For each module found, …

C++ 核心编程

一、 内存分区模型 C程序在执行时&#xff0c;将内存大方向划分为4个区域 代码区&#xff1a;存放函数体的二进制代码&#xff0c;由操作系统进行管理 全局区&#xff1a;存放全局变量和静态变量以及常量 栈区&#xff1a;由编译器自动分配释放&#xff0c;存放函数的参数值、…

gensim conherence model C_V 值与其他指标负相关BUG

在我用gensim3.8.3 conherence model分析京东评论主题模型时&#xff0c; C_V 与npmi、u_mass出现了强烈的皮尔逊负相关&#xff1a; 这些地方也反映了类似问题&#xff1a; https://github.com/dice-group/Palmetto/issues/12 https://github.com/dice-group/Palmetto/issue…

测等保2.0——安全区域边界

一、前言 今天我们来说说安全区域边界&#xff0c;顾名思义&#xff0c;安全区域边界就是保障网络边界处&#xff0c;包括网络对外界的边界和内部划分不同区域的交界处&#xff0c;我们的重点就是查看这些边界处是否部署必要的安全设备&#xff0c;包括防火墙、网闸、网关等安…

hypef 九、日志

教程&#xff1a;Hyperf hyperf结合monolog是属于低耦合&#xff0c;通过配置文件和调用工厂类设置参数设置实例。 monolog详见Monolog 修改_lsswear的博客-CSDN博客&#xff0c;理解monolog原理之后若有对应的报错比较好改。 一、使用LoggerFactory #/config/autoload/log…

git相关

gerrit用户指南&#xff1a; 资料&#xff1a;Gerrit 用户指南 gerrit-user-guide 上述有介绍如何review&#xff0c;review并非修改代码之后如何重新提交等操作 jenkins介绍 Jenkins详细教程 - 知乎 一、jenkins是什么&#xff1f; Jenkins是一个开源的、提供友好操作界…

使用Ensp配置DHCP协议

如何使用Ensp配置DHCP协议&#xff0c;为PC自动分配IP地址 什么是DHCP&#xff1f; Dynamic Host Configuration Protocol&#xff0c;动态主机配置协议&#xff0c;简单理解为自动分配IP地址&#xff0c;有了这个协议就不用手动配置IP地址了&#xff0c;如图 思路 给路由…

iptable的备份和还原

1、写在命令行当中的都是临时配置 2、把我们的规则配置在服务的文件当中&#xff0c;形成永久生效&#xff0c; iptables -save > /opt/test&#xff08;自设&#xff09; #进行备份 /etc/sysconfig/iptables #默认配置文件 写在配置文件中才会永久生效 iptables-r…

Docker如何使用

一、Docker安装 命令:curl -fsSL https://get.docker.com | bash -s docker --mirror Aliyun 1.1 Docker技术的三大核心概念 镜像:类似一个root系统文件 容器&#xff1a;镜像运行的实体&#xff0c;可以被创建、启动、停止、删除、暂停等 仓库&#xff1a;保存镜像 二、镜…