Vue笔记_01双向数据绑定原理

news2025/7/17 9:25:39

[1]什么叫双向数据绑定?

  • 视图中的数据发生了变化,data中的数据也要对应改变;
  • data中的数据发生了变化,视图上的数据也要对应改变;

[2]双向绑定原理

vue 双向数据绑定是通过 数据劫持 结合 发布订阅模式的方式来实现的, 也就是说数据和视图同步,数据发生变化,视图跟着变化,视图变化,数据也随之发生改变;

vue2.x实现双向绑定

在vue2.x中数据双向绑定的核心是使用 Object.defineProperty(对象,key,{get(),set()})方法来给对象的属性添加getset方法实现的!

Object.defineProperty对象(es6新增)
语法
Object.defineProperty(obj,key,{
  enumerable:false , // 控制属性是否可以枚举,默认值为false
  writable:false, // 控制属性是否可以被修改,默认值为false
  configurable:false, // 控制属性是否可以被删除,默认值为false
  // 当获取该属性时就会调用get方法 ,返回值为获取到的值
  get(){
    return xxx
  }
  // 当给该属性进行赋值时就会调用set方法
  set(val){
  }
})
举例说明

需求:现在有如下代码,希望person对象存在一个age属性,属性值为number变量的值且 number值与age属性值能够实时双向绑定

let number = 18
let person = {
  name:'chaochao',
  sex:'女'
}

实现:

let number = 20
let person = {
  name:'chaochao',
  sex:'女'
}
Object.defineProperty(person,'age',{
  get(){
    console.log('@@@获取age')
    return number
  },
  set(val){
    console.log('@@@修改age')
    number = val
  }
})

结果:
在这里插入图片描述

踩坑 - 给某属性进行双向绑定

在此示例中给data对象的name属性通过defineProperty进行 读取/赋值;

let data = {
  name: 'chaochao'
}
Object.defineProperty(data, 'name', {
  get () {
    // [2] 若是在返回时直接通过点语法获取会再次调用get方法 -> 递归却没有递归边界---死循环
    return data.name
  },
  set (value) {
    data.name = value
  }
})
// [1]在此处获取了data的name属性值,就会调用name属性的get方法
data.name

改正:

let data = {
  name: 'chaochao'
}
let _name = data.name
Object.defineProperty(data, 'name', {
  get () {
    return _name
  },
  set (value) {
    data.name = value
  }
})
// [1]在此处获取了data的name属性值,就会调用name属性的get方法
data.name
踩坑 - 给对象的所有属性进行双向绑定
  let data = {
  name: 'chaochao',
  sex:'女',
  say:''
}
let _name = data.name
for(let key in data){
  Object.defineProperty(data, key, {
    get () {
      return data[key] // 无限调用
    },
    set (str) {
      data[key] = str // error: Maximum call stack size exceeded
    }
})
}

改正:

let data = {
  name:'chaochao',
  sex:'女',
  say:''
}
let _name = data.name
Object.keys(data).forEach((key) => {
  defineProperty(data, key, data[key])
})
function defineProperty(obj,key,val) {
  Object.defineProperty(obj, key, {
    get () {
      console.log('@@@get', val)
      return val
    },
    set (str) {
      console.log('@@@set', val,str)
      val = str
    }
})
}

在这里插入图片描述

vue2.x实现双向绑定

在通过new关键字实例化对象时,传入的配置项中添加了 data属性

const vm = new Vue({
   el:'#app',
   data:{
     name:'chaochao',
     sex:'女',
     say:'hello word'
   }
 })

vue会将data存在在实例化对象的_data属性中;
在这里插入图片描述

const data = {
  name:'chaochao',
  sex:'女',
  say:'hello word'
}
const vm = new Vue({
  el:'#app',
  data
})
console.log(data==vm._data) // true

将data中的数据平铺在vue实例化对象身上
在这里插入图片描述
使用Object.defineProperty实现数据代理。
模拟数据代理过程

// 模拟配置项data
const data = {
  name:'chaochao',
  sex:'女',
  say:'hello word'
}

// 模拟vue实例化对象 代码中更新的是_data中的数据,视图上更改的是vue实例化对象身上的数据
const vm ={
  _data:data, // 存储数据
  ...data // 数据劫持-> 更新试图
}
for(let key in vm._data){
  Object.defineProperty(vm._data, key , {
    //  当获取属性值时返回vue实例化对象身上的对应的属性的属性值
    get(){
      return vm[key]
    },
    // 当修改属性值时-> 将vue实例化对象身上对应的属性值一起修改
    set(val){
      vm[key] = val
    }
  })
}

在这里插入图片描述

vue2.x实现双向绑定缺点

  • 对象:只能给对象中已经存在的属性进行双向绑定!
    • 新增属性、删除属性,视图不会更新
  • 数组:直接通过下标修改数组,视图不会更新

所以在vue2.x中有时会出现双向绑定失败(实例化对象中的数据改变了,但是视图上并没有重新渲染)的现象-可以使用 $set 解决

vue3.x实现双向绑定

在vue2.x中数据双向绑定的核心是使用new Proxy(对象,{get(),set()})方法来给对象的属性添加getset方法实现的!

Proxy构造函数(es6新增)

语法

let data = {
  name:'chaochao',
  sex:'女',
  say:''
}
//data为源数据 p为代理数据
const p = new Proxy(data,{
  // 拦截读取属性值  ->  当读取p身上的属性时就会触发该方法,该函数return的值就是读取到的值
  // target:源数据; prop:key值
  get (target, prop) {
    console.log('@@@读取数据了')
    return target[prop] // 默认返回的是源数据的值
  },
  // 拦截设置属性值或添加新属性 -> 当修改p身上的属性的属性值或给p添加新属性时就会触发该方法
  set (target, prop, value) {
    console.log('@@@修改数据')
    target[prop] = value // 当修改了代理数据就去修改源数据的值
  },
  // 拦截删除属性 -> 当删除p身上的属性时就会触发该方法
  deleteProperty (target, prop) {
    console.log('@@@删除数据 ')
    return delete[prop] // 当删除了代理数据的属性就删除源数据的属性并返回是否删除成功
  }
})

Reflect对象

我们可以通过Reflect对象去增删改查对象中的属性
语法

let data = {
  name:'chaochao',
  sex:'女',
  say:''
}
// 新增属性
console.log(Reflect.set(data,'age', 18)) // true
// 获取属性
console.log(Reflect.get(data,'name'), Reflect.get(data,'age')) // chaochao, 18
// 修改属性
console.log(Reflect.set(data,'name','niuniu')) // true
// 删除属性
console.log(Reflect.deleteProperty(data,'sex')) // true
// 查看对象
console.log(data) // {name: 'niuniu', say: '', age: 18}

优点是容错性强!

vue3.x双向绑定

 new Proxy(data,{
    get (target, prop) {
      return Reflect.get(target,prop)
    },
    set (target, prop, value) {
      Reflect.set(target,prop,value)
    },
    deleteProperty (target, prop) {
      return Reflect.deleteProperty(target, prop)
    }
})
  • 通过Proxy(代理): 拦截对象中任意属性的变化, 包括:属性值的读写、属性的添加、属性的删除等。
  • 通过Reflect(反射): 对源对象的属性进行操作。

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

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

相关文章

Pip版本问题导致Python模块安装失败

文章目录起因解决方案前言输入 %APPDATA%创建 pip.ini终端执行命令安装包结语起因 今天在视频号平台看到有小姐姐直播讲爬虫技术,我一看这不是挺简单的吗?于是我就想到自己四年前的今天刚进大一的时候最开始学的爬虫技术,后来因为一些事情跑…

怎么证明前端数据加密的三种方式

导读:前端最常见的三大加密方式https,SSH和MD5,这篇文章带你走进三大加密方式的原理对比。 1.https 1.1原理 A.就是在http加入SSL层,是http安全的基础; B.htts协议是在http基础上加了SSL协议; C.使用443端口,http是80…

2022CTF培训(三)windowslinux安卓平台调试机制原理

附件下载链接 windows平台调试机制原理 手动编写一个简易调试器 创建待调试进程 使用 CreateProcess 函数创建待调试进程,创建时指定 dwCreationFlags 参数为 DEBUG_ONLY_THIS_PROCESS 将会告诉操作系统我们需要让当前调用者(线程)接管所…

(十)centos7案例实战——实现nginx代理访问redis服务

前言 本节内容是关于实现nginx代理访问redis服务,由于在实际生产开发环境中,我们并不想将我们的中间键服务暴露在公网环境中,或者只能在内网环境中使用,例如本节内容,我们将redis安装到本地环境,但是又有需…

链表中倒数第k个结点、反转链表、合并两个排序的链表、树的子结构、删除链表中重复的结点

文章目录1、链表中倒数第k个结点2、反转链表3、合并两个排序的链表4、树的子结构5、 二叉树的镜像6、删除链表中重复的结点1、链表中倒数第k个结点 本题考点: 链表,前后指针的使用,边界条件检测 牛客链接 题目描述: 输入一个链表…

JVM【八股文】

JVM【八股文】 JVM内存区域划分 程序计数器栈堆方法区 一块大的区域,需要根据功能,来划分不同的小区域。 JVM内存是从操作系统里申请来的,之后堆这部分区域进行了划分。 1.程序计数器 内存中最小的区域,保存了下一条要执行指令…

android-加壳加固

title: android-加壳加固 categories: Android tags: [android, 加壳, 加固, 混淆] date: 2022-06-20 18:00:23 comments: false mathjax: true toc: true android-加壳 前篇 Android之Apk加壳 - https://blog.csdn.net/LVXIANGAN/article/details/84956476Android动态加载Dex…

李沐论文精度系列之七:Two-Stream双流网络、I3D

文章目录一、双流网络1.1 前言1.2 网络结构1.3 光流(Optical flow)1.3.1 什么是光流1.3.2 如何利用光流1.3.3 双向光流(Bi-directional optical flow)1.3.4 光流的局限性及和对应的预处理(抽取)方式1.3.5 视频模型测试1.4 实验1.4…

✿✿✿JavaScript基本语法一

目 录 1.js的发展史(闲聊版) 2.浏览器分成两部分:渲染引擎和 JS 引擎 3.js与html的关系以及结合方式 (1)js与html的关系 (2)js与html结合方式 4.JavaScript注释 5.js中的基本数据类型 6.js中的变量 7.运算符(自动类型转…

9.前端笔记-CSS-盒子模型-border和padding

页面布局的三大核心: 盒子模型浮动定位 1、盒子模型 1.1 盒子模型组成 盒子模型本质还是一个盒子,包括边框border、外边距margin、内边距padding和实际内容content 1.1.1 边框border 组成 组成:颜色border-color、边框宽度border-wid…

518. 零钱兑换 II【完全背包:求组合数】

518. 零钱兑换 II 给你一个整数数组 coins 表示不同面额的硬币,另给一个整数 amount 表示总金额。 请你计算并返回可以凑成总金额的硬币组合数。如果任何硬币组合都无法凑出总金额,返回 0 。 假设每一种面额的硬币有无限个。 题目数据保证结果符合 32 位…

C++11 右值,右值引用,移动构造,移动赋值

目录 一、左值,左值引用,右值,右值引用的相关概念: 1. 什么是左值,什么是左值引用? 2. 什么是右值,什么是右值引用? 3. 右值的属性是右值,右值引用的属性是左值 4. …

棒子老虎鸡-第12届蓝桥杯Scratch选拔赛真题精选

[导读]:超平老师计划推出Scratch蓝桥杯真题解析100讲,这是超平老师解读Scratch蓝桥真题系列的第86讲。 蓝桥杯选拔赛每一届都要举行4~5次,和省赛、国赛相比,题目要简单不少,再加上篇幅有限,因此我精挑细选…

研究生有限元仿真应用中存在的问题与对策

作者:尚晓江 导读:有限元分析软件作为计算工具,在科研和工程领域都有广泛应用,而多数用户是在研究生阶段开始接触和使用这些计算软件的。本文以ANSYS结构分析为例,对现阶段研究生应用有限元分析软件的现状和存在的问题…

无人机设计仿真--在Isight平台上进行的基于CST参数化+Xfoil的无人机翼型优化

作者:Graychen 一、工程背景 翼型的选型和设计是飞行器气动设计中的一项基础性工作,翼型对飞行器的气动性能具有根本性的影响。现在高性能飞行器已不再从翼型库中选择适用翼型后直接使用,而是以现有翼型作为基准翼型进行气动优化&#xff…

java基本语法 下

目录 运算符 运算符:算术运算符 运算符:赋值运算符 运算符:比较运算符 运算符:逻辑运算符 运算符:三元运算符 运算符的优先级 程序流程控制 概念 顺序结构 if-else结构 switch-case结构 循环结构 循环结构…

Unity视差贴图多实现对比和改进

视差贴图多种实现方式对比和改进视差贴图视差映射陡峭视差映射视差遮蔽映射迭代视差映射-kerry视差贴图 参考 与法线贴图相同,可以模拟出物体得深度感,同时它得改进是能够随着视角得偏移显示不同得深度感,使得显示更加真实。 由于采样高度…

代码随想录刷题| 多重背包理论基础、背包问题的总结

目录 多重背包理论基础 多重背包的问题 多重背包的解法 多重背包的代码 背包问题的总结 01背包 完全背包 多重背包 多重背包理论基础 多重背包的问题 有N种物品和一个容量为V 的背包。第i种物品最多有Mi件可用,每件耗费的空间是Ci ,价值是Wi 。…

单身福利专场, Python采集某相亲网站美女数据

前言 嗨喽~大家好呀,这里是魔王呐 ❤ ~! 现在,广大年轻人到了一定年纪,一定会引来父母的念叨 不是让相亲就是让结婚的,与其父母念叨,不如自己找一个 到时候问起来,就说再接触呢~~ 今天我们就来用python…

NestJS学习:使用session实现登录验证

参考 大佬的视频教程:nestjs session案例 大佬的博客地址:小满nestjs(第九章 nestjs Session) 在学习某些知识时如果有大佬的视频教程与文档真的是太爽了,能够学习到好多新知识。 nest后台 session session 是服务…