• linkedu视频
  • 平面设计
  • 电脑入门
  • 操作系统
  • 办公应用
  • 电脑硬件
  • 动画设计
  • 3D设计
  • 网页设计
  • CAD设计
  • 影音处理
  • 数据库
  • 程序设计
  • 认证考试
  • 信息管理
  • 信息安全
菜单
linkedu.com
  • 网页制作
  • 数据库
  • 程序设计
  • 操作系统
  • CMS教程
  • 游戏攻略
  • 脚本语言
  • 平面设计
  • 软件教程
  • 网络安全
  • 电脑知识
  • 服务器
  • 视频教程
  • dedecms
  • ecshop
  • z-blog
  • UcHome
  • UCenter
  • drupal
  • WordPress
  • 帝国cms
  • phpcms
  • 动易cms
  • phpwind
  • discuz
  • 科汛cms
  • 风讯cms
  • 建站教程
  • 运营技巧
您的位置:首页 > CMS教程 >建站教程 > vue2&vue3数据响应式原理分析及手动实现(实例详解)

vue2&vue3数据响应式原理分析及手动实现(实例详解)

作者:站长图库 字体:[增加 减小] 来源:互联网

站长图库向大家介绍了vue数据响应式,原理分析等相关知识,希望对您有所帮助

本篇文章给大家带来了vue2&vue3数据响应式原理分析及手动实现的相关知识,数据响应式视图跟数据是自动更新的,数据更新的时候视图是自动的更新的追踪数据的变化,希望对大家有帮助。


vue2&vue3数据响应式原理分析及手动实现(实例详解)


数据响应式

视图跟数据是自动更新的,数据更新的时候视图是自动的更新的

追踪数据的变化,在读取数据或者设置数据的时候能够做一些劫持的一些操作

vue2 使用defineProperty

vue3 改用Proxy


使用defineProperty

如何追踪变化

var obj = {}var age Object.defineProperty(obj, 'age', {    get: function() {        consoel.log('get age ...')        return age    },    set: function(val) {        console.log('set age ...')        age = val    }})obj.age =100 //set age ...console.log(obj.age) //get age ...

对象obj在取age属性的时候会调用数据劫持的get方法

在给age属性赋值的时候会调用set方法

那怎么使用Object.defineProperty实现一个数据响应式呢

function defineReactive(data) {  if (!data || Object.prototype.toString.call(data) !== '[object Object]')    return;  for (let key in data) {    let val = data[key];    Object.defineProperty(data, key, {      enumerable: true, //可枚举      configurable: true, //可配置      get: function() {        track(data, key);        return val;      },      set: function() {        trigger(val, key);      },    });    if (typeof val === "object") {      defineReactive(val);    }  }}function trigger(val, key) {  console.log("sue set", val, key);}function track(val, key) {  console.log("sue set", val, key);}const data = {  name:'better',  firends:['1','2']}defineReactive(data)console.log(data.name)console.log(data.firends[1])console.log(data.firends[0])console.log(Object.prototype.toString.call(data))

这个函数defineReactve用来对Object.defineProperty进行封装,从函数名可以看出,起作用就是定义一个响应式数据,封装后只需要传递data,key和val就行

每当从data中读取key的时候触发track函数,往data的key中设置数据时,set函数中的trigger函数触发

数组的响应式

我们通过Array原型上的方法来改变数组的内容不会触发getter和setter

整理发现Array原型中可以改变数组自身内容的方法有7个,分别push pop shift unshift splice sort reverse

vue2 改写了这这7种方法

实现方式:

以Array.propertype为原型创建一个arrayMethods对象,再使用Object.setPropertypeOf(o, arryMethods)将o的__proto__指向arrayMethods


vue2&vue3数据响应式原理分析及手动实现(实例详解)

如何收集依赖

使用

<template><p>{{name}}</p></template>

该模板中使用数据 name, 我们要观察数据, 当数据的属性发生变化的时候, 可以通知哪些使用的地方

这就是我们要先收集依赖,即把用到数据name的地方收集起来,然后等数据变化的时候,把之前收集好的依赖循环触发一遍,总结来说就是getter中收集依赖,在setter中触发依赖

使用proxy

Proxy对象用于创建一个对象的代理, 从而实现基本操作的拦截和定义(如属性查找、赋值、枚举、函数掉用等)

const p = new Proxy(target, handler)

target

要使用 Proxy 包装的目标对象(可以是任何类型的对象,包括原生数组,函数,甚至另一个代理)。

handler

一个通常以函数作为属性的对象,各属性中的函数分别定义了在执行各种操作时代理 p 的行为。

reflect是一个内置对象, 他提供拦截javascript操作的方法, 这些方法和Proxy handlers相同

Reflect.set将值分配给属性的函数。返回一个Boolean 如果更新成功则返回true

Reflect.get获取对象身上某个属性的值,类似target[name]

如何实现劫持

const dinner = {  meal:'111'}const handler = {  get(target, prop) {    console.log('get...', prop)    return Reflect.get(...arguments)  },  set(target, key, value) {    console.log('get...', prop)    console.log('set',key,value)    return Reflect.set(...arguments)  }}const proxy = new Proxy(dinner, handler)console.log(proxy.meal)console.log(proxy.meal)

代码中dinner 对象代理到handler上

跟defineProperty区别

defineProperty的属性需要遍历才能监管所有属性

使用proxy可以将对象所有属性进行代理

用proxy实现一个模拟响应式

function reactive(obj) {  const handler = {    get(target, prop, receiver) {      track(target, prop);      const value =  Reflect.get(...arguments);      if (typeof value === 'Object') {        reactive(value)      } else {        return value      }    },    set(target,key, value, receiver) {      trigger(target,key, value);      return Reflect.set(...arguments);    },  };  return new Proxy(obj,handler)}function track(data, key) {  console.log("sue set", data, key);}function trigger(data, key,value) {  console.log("sue set", key,':',value);}const dinner = {  name:'haochi1'}const proxy  = reactive(dinner)proxy.nameproxy.list = []proxy.list.push(1)

执行后自动打印


vue2&vue3数据响应式原理分析及手动实现(实例详解)

分享到:QQ空间新浪微博腾讯微博微信百度贴吧QQ好友复制网址打印

您可能想查找下面的文章:

  • vue2&vue3数据响应式原理分析及手动实现(实例详解)

相关文章

  • dedecms文章审核后给会员增加积分或者金币的办法
  • PhotoShop绘制反光水晶玻璃球按钮教程
  • PHP指定范围内且不重复的随机值方法
  • PS制作超酷黑白像素文字效果
  • DEDE建站内部SEO优化
  • Javascript怎么实现四位随机验证码
  • JS数组如何删除指定元素
  • Photoshop绘制任天堂Wii游戏手柄
  • 手动触发 Lazyload 显示懒加载的图片
  • 浅析网页与小程序间怎么进行通信

文章分类

  • dedecms
  • ecshop
  • z-blog
  • UcHome
  • UCenter
  • drupal
  • WordPress
  • 帝国cms
  • phpcms
  • 动易cms
  • phpwind
  • discuz
  • 科汛cms
  • 风讯cms
  • 建站教程
  • 运营技巧

最近更新的内容

    • 安全狗卸载网站打不开/iis启动不了解决方法
    • 聊聊node中怎么使用Nest.js 实现简易版请求监控
    • 手把手带你了解Vue中怎么使用filters过滤器?2种用法浅析
    • renderjs有什么用?聊聊uniapp中用renderjs的一些细节
    • 提高mysql千万级大数据SQL查询优化30条经验(Mysql索引优化注意)
    • PHP怎么实现评论回复功能
    • 详解thinkphp6后台添加google登录验证
    • 利用图层样式及素材制作漂亮的火焰字
    • PHP过滤HTML标签代码方法
    • vue3为什么快?vue3的效率提升主要在哪方面?

关于我们 - 联系我们 - 免责声明 - 网站地图

©2020-2025 All Rights Reserved. linkedu.com 版权所有